牛客周赛 Round 28 解题报告 | 珂学家 | 组合数学 + 离散化&树状数组

小红的新周赛

https://ac.nowcoder.com/acm/contest/73239/A


前言

alt


整体评价

还是E稍微有点意思,新周赛好像比预期要简单一些, ^_^.


欢迎关注

珂朵莉 牛客周赛专栏

珂朵莉 牛客小白月赛专栏


A. 小红的新周赛

思路: 模拟

#include <bits/stdc++.h>

using namespace std;

int main() {
    int res = 0;
    for (int i = 0; i < 6; i++) {
        int v;
        cin >> v;
        res += v;
    }
    cout << res << endl;
    return 0;
}

B. 小红的字符串

思路: 计数+模拟

引入 26 * 26的状态,进行计数

这有个好处,就是天然排序,避免大内存存字符串并排序

#include <bits/stdc++.h>

using namespace std;

int main() {
    
    // 26 * 26天然保序
    int cnt[26][26] = {0};
    
    string s;
    cin >> s;
    int n = s.length();
    for (int i = 0; i < n - 1; i++) {
        int p1 = s[i] - 'a';
        int p2 = s[i + 1] - 'a';
        cnt[p1][p2]++;
    }
    
    for (int i = 0; i < 26; i++) {
        for (int j = 0; j < 26; j++) {
            string ts = "";
            ts.push_back((char)(i + 'a'));
            ts.push_back((char)(j + 'a'));
            for (int t = 0; t < cnt[i][j]; t++) {
                cout << ts << endl;
            }
        }
    }
    
}

C. 小红的炸砖块

思路: 模拟

引入保存每列高度的数组,然后模拟即可

#include <bits/stdc++.h>

using namespace std;

int main() {
    
    int n, m, k;
    cin >> n >> m >> k;
    
    vector<int> cols(m, n);
    for (int i = 0; i < k; i++) {
        int r, c;
        cin >> r >> c;
        if (cols[c - 1] >= n - r + 1) {
            cols[c - 1]--;
        }
    }
    
    for (int i = 0; i < n; i++) {
        string r;
        for (int j = 0; j < m; j++) {
            r.push_back(cols[j] >= n - i ? '*' : '.');
        }
        cout << r << endl;
    }
    
    return 0;
}

D. 小红统计区间(easy)

思路: 滑窗

非常典的一道滑窗题,双指针维护即可

#include <bits/stdc++.h>

using namespace std;

using int64 = long long;

int main() {
    
    int n;
    int64 k;
    cin >> n >> k;
    
    vector<int64> pre(n + 1, 0);
    vector<int> arr(n);
    for (int i = 0; i < n; i++) {
        cin >> arr[i];
        pre[i + 1] = pre[i] + arr[i];
    }
    
    int64 res = 0LL;
    int j = 0;
    for (int i = 0; i < n; i++) {
        while (j <= i && pre[i + 1] - pre[j] >= k) {
            j++;
        }
        res += j;
    }
    cout << res << endl;
    
    return 0;
}

E. 小红的好数组

思路: 找规律 + 组合数学

case给的非常良心

可以分类讨论,大概有4种类似的序列

arr1 = [偶数,偶数,偶数,偶数,偶数,偶数, ...]

arr2 = [奇数,奇数,偶数,奇数,奇数,偶数,...]

arr3 = [奇数,偶数,奇数,奇数,偶数,奇数,...]

arr4 = [偶数,奇数,奇数,偶数,奇数,奇数,...]

奇数/偶数的分布,呈现强烈的规律

最终为这4种情况的组合方案和

#include <bits/stdc++.h>

using namespace std;

using int64 = long long;

const int64 mod = (long)1e9 + 7;

int64 ksm(int64 b, int64 v) {
    int64 r = 1LL;
    while (v > 0) {
        if (v % 2 == 1) {
            r = r * b % mod;
        }
        v /= 2;
        b = b * b % mod;
    }
    return r;
}

int main() {
    
    int n, k;
    cin >> n >> k;
    
    int k2 = k / 2, k1 = k - k2;
    
    int64 r1 = ksm(k2, n);
    int64 r2 = ksm(k2, n/3) * ksm(k1, n - n/3) % mod;
    int64 r3 = ksm(k2, (n+1)/3) * ksm(k1, n - (n+1)/3) % mod;
    int64 r4 = ksm(k2, (n+2)/3) * ksm(k1, n - (n+2)/3) % mod;
    
    int64 res = (r1 + r2 + r3 + r4) % mod;
    cout << res << endl;
    
    return 0;
}

F. 小红统计区间(hard)

思路: 离散化+树状数组

也是一道非常典的题

因为存在负数,所以滑窗的基础已经被破坏了

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.*;

public class Main {

    static class BIT {
        int n;
        int[] arr;
        public BIT(int n) {
            this.n = n;
            this.arr = new int[n + 1];
        }

        int query(int p) {
            int res = 0;
            while (p > 0) {
                res += arr[p];
                p -= p & -p;
            }
            return res;
        }

        void update(int p, int d) {
            while (p <= n) {
                arr[p] += d;
                p += p & -p;
            }
        }
    }

    public static void main(String[] args) {
        AReader sc = new AReader();

        int n = sc.nextInt();
        long k = sc.nextLong();

        long[] arr = new long[n];
        long[] pre = new long[n + 1];
        for (int i = 0; i < n; i++) {
            arr[i] = sc.nextLong();
            pre[i + 1] = pre[i] + arr[i];
        }

        // 进行离散化
        TreeSet<Long> ids = new TreeSet<>();
        for (long v: pre) {
            ids.add(v);
        }
        int ptr = 0;
        TreeMap<Long, Integer> hp = new TreeMap<>();
        for (long kv: ids) {
            hp.put(kv, ++ptr);
        }

        BIT bit = new BIT(ptr);
        bit.update(hp.get(0l), 1);

        long res = 0;
        for (int i = 0; i < n; i++) {
            long p = pre[i + 1];
            // p - x >= k
            // x <= p - k
            Map.Entry<Long, Integer> ent = hp.floorEntry(p - k);
            if (ent != null) {
                res += bit.query(ent.getValue());
            }
            bit.update(hp.get(p), 1);
        }

        System.out.println(res);

    }

    static
    class AReader {
        private BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
        private StringTokenizer tokenizer = new StringTokenizer("");
        private String innerNextLine() {
            try {
                return reader.readLine();
            } catch (IOException ex) {
                return null;
            }
        }
        public boolean hasNext() {
            while (!tokenizer.hasMoreTokens()) {
                String nextLine = innerNextLine();
                if (nextLine == null) {
                    return false;
                }
                tokenizer = new StringTokenizer(nextLine);
            }
            return true;
        }
        public String nextLine() {
            tokenizer = new StringTokenizer("");
            return innerNextLine();
        }
        public String next() {
            hasNext();
            return tokenizer.nextToken();
        }
        public int nextInt() {
            return Integer.parseInt(next());
        }

        public long nextLong() {
            return Long.parseLong(next());
        }

//        public BigInteger nextBigInt() {
//            return new BigInteger(next());
//        }
        // 若需要nextDouble等方法,请自行调用Double.parseDouble包装
    }

}

写在最后

alt

牛客周赛解题报告系列 文章被收录于专栏

牛客周赛,定位求职笔试真题,倾向于面试算法

全部评论
为什么我看有人d题用二分在写
2 回复 分享
发布于 01-14 21:23 浙江
先点赞后看
1 回复 分享
发布于 01-15 20:41 广东

相关推荐

沉淀一会:**圣经 1.同学你面试评价不错,概率很大,请耐心等待;2.你的排名比较靠前,不要担心,耐心等待;3.问题不大,正在审批,不要着急签其他公司,等等我们!4.预计9月中下旬,安心过节;5.下周会有结果,请耐心等待下;6.可能国庆节前后,一有结果我马上通知你;7.预计10月中旬,再坚持一下;8.正在走流程,就这两天了;9.同学,结果我也不知道,你如果查到了也告诉我一声;10.同学你出线不明朗,建议签其他公司保底!11.同学你找了哪些公司,我也在找工作。
点赞 评论 收藏
分享
hso_:哈哈哈哈哈哈我没offer一样在同一道题开喷了
投递深圳同为数码等公司10个岗位
点赞 评论 收藏
分享
评论
16
1
分享
牛客网
牛客企业服务