zoukankan      html  css  js  c++  java
  • Codeforces 1585E Frequency Queries

    题目大意

    给定一棵 \(n(n\leq 10^6)\) 个点的树,结点 \(u\) 的点权为 \(a_u(1\leq a_u\leq n)\)\(q(q\leq 10^6)\) 次询问,每次询问 \(u,l,k\),表示从 \(u\) 到根的树链上所有点权拿出来,把点权按出现次数从小到大排序再去重,去掉出现次数少于 \(l\) 的点权,剩下的数组里的第 \(k\) 个点权是多少。答案可能有多解,输出任意一个,若无解,输出 \(-1\)

    题解

    首先询问可以离线,所以可以每遍历到一个点就处理关于这个点的所有询问。

    \(cnt[x]\) 表示点权 \(x\) 出现的数目。在DFS的过程中很容易可以维护当前点 \(u\) 到根的路径上所有的点权的 \(cnt[x]\)

    \(p\)\(1\sim n\) 的一个排列,表示点权 \(1\sim n\) 按在 \(u\) 到根的路径上出现的次数从小到大排序的结果,设 \(p^{-1}\) 表示 \(p\) 的逆排列,即 \(p^{-1}[p[i]]=i\),设 \(lb_x\) 表示第一个出现次数大于等于 \(x\) 的点权在 \(p\) 中的下标。我们只要维护 \(cnt,p,p^{-1},lb\) 数组,最后询问 \(u,l,k\) 的答案即为 \(p[lb[l]+k-1]\)

    当DFS进入一个点 \(u\) 时,我们令 \(x:=a[u]\)。此时 \(p\) 一定是被分为若干段,每一段内的点权出现的次数相等,段与段之间点权出现的次数相差1。那么我们需要考虑 \(++cnt[x]\) 时对 \(p,p^{-1},lb\) 的影响。我们可以把 \(p\)\(x\) 的位置和 \(x\) 所在段的最后一个数的位置互换,然后让当前段的最后一个位置变成下一段的第一个位置即可,即:

    int x = a[u];
    int p1 = pt[x], p2 = lb[cnt[x] + 1] - 1;
    swap(p[p1], p[p2]);
    int pt1 = x, pt2 = p[pt[x]];
    swap(pt[pt1], pt[pt2]);
    --lb[cnt[x] + 1]; ++cnt[x];
    

    当退出 \(u\) 结点时需要恢复影响:

    --cnt[x]; ++lb[cnt[x] + 1];
    swap(pt[pt1], pt[pt2]);
    swap(p[p1], p[p2]);
    

    时间复杂度 \(O(n+q)\)

    感觉这个单点 \(O(1)\) 修改挺巧妙的。

    Code

    #include <bits/stdc++.h>
    using namespace std;
    
    template<typename elemType>
    inline void Read(elemType& T) {
        elemType X = 0, w = 0; char ch = 0;
        while (!isdigit(ch)) { w |= ch == '-';ch = getchar(); }
        while (isdigit(ch)) X = (X << 3) + (X << 1) + (ch ^ 48), ch = getchar();
        T = (w ? -X : X);
    }
    
    const int maxn = 1000010;
    
    vector<int> G[maxn];
    struct Node { int id, l, k; };
    vector<Node> ask[maxn];
    int a[maxn], ans[maxn], cnt[maxn], p[maxn], pt[maxn], lb[maxn];
    int T, n, q;
    
    void DFS(int u, int fa) {
        int x = a[u];
        int p1 = pt[x], p2 = lb[cnt[x] + 1] - 1;
        swap(p[p1], p[p2]);
        int pt1 = x, pt2 = p[pt[x]];
        swap(pt[pt1], pt[pt2]);
        --lb[cnt[x] + 1]; ++cnt[x];
        for (auto node : ask[u]) {
            if (lb[node.l] + node.k - 1 > n) ans[node.id] = -1;
            else ans[node.id] = p[lb[node.l] + node.k - 1];
        }
        for (auto v : G[u]) {
            if (v == fa) continue;
            DFS(v, u);
        }
        --cnt[x]; ++lb[cnt[x] + 1];
        swap(pt[pt1], pt[pt2]);
        swap(p[p1], p[p2]);
    }
    
    int main() {
        Read(T);
        while (T--) {
            Read(n);Read(q);
            lb[0] = 1;
            for (int i = 1;i <= n;++i) { G[i].clear(); p[i] = pt[i] = i; lb[i] = n + 1; cnt[i] = 0; }
            for (int i = 1;i <= n;++i) { Read(a[i]); ask[i].clear(); }
            for (int i = 2;i <= n;++i) { int u; Read(u); G[u].push_back(i); G[i].push_back(u); }
            for (int i = 1;i <= q;++i) {
                int v, l, k;
                Read(v);Read(l);Read(k);
                ask[v].push_back((Node) { i, l, k });
            }
            DFS(1, 0);
            for (int i = 1;i <= q;++i)
                printf("%d ", ans[i]);
            printf("\n");
        }
    
        return 0;
    }
    
  • 相关阅读:
    基础练习 龟兔赛跑预测
    SQL查询
    通讯录管理系统
    信号分解
    开源框架 openFrameworks
    开博了
    AES加密原理和AOE工程实践
    一文读懂AOE到底是什么!
    AoE 搭档 TensorFlow Lite ,让终端侧 AI 开发变得更加简单。
    AOE工程实践-银行卡OCR里的图像处理
  • 原文地址:https://www.cnblogs.com/AEMShana/p/15731704.html
Copyright © 2011-2022 走看看