题目描述

给定\(n\)个数$a_i$,起初第\(i\)个数在第\(i\)个集合。有三种操作(共\(m\)次):

1 $u$ $v$ 将第$u$个数和第$v$个数所在集合合并

2 $u$ 将第$u$个数所在集合所有数加1

3 $u$ $k$ $x$ 问$u$所在集合有多少个数模$2^k$余$x$。

数据范围:\(n,m \le 500000,a_i \le 10^9, 0 \le k \le 30\)。

简要题解

显然此题可以用set加启发式合并在\(O(n \log ^2 n)\)时间复杂度解决本题,但此题时限1.5s,必须使用一个log的做法。事实上,这是一道十分套路的Trie树题目。

首先用并查集维护连通性。

下面先考虑操作3,这相当于询问低$k$位二进制固定时集合中元素个数,可以用Trie树,维护一个子树中终结结点有多少个即可。

对于加1操作,可以在Trie树上打标记,类似线段树进行pushDown标记下传。此题pushDown函数很新颖(之前没写过这样的pushDown),详见代码。(队友指出,此题也可以暴力更新Trie树,至多交换\(\log 10^9\)个结点;不过如果每次加的数不是1的话就必须打标记了。)

对于合并操作,类似线段树合并。由于初始时\(n\)个数共需要\(O(n \log 10^9)\)个结点,而花费O(1)的时间会将总结点数减1,故Trie树合并的总时间复杂度也为\(O(n \log 10^9)\)。关于线段树合并,可以做这道入门题练手:Codeforces Gym 101194G(2016EC Final)

总时间复杂度\(O((n+q) \log 10^9)\)。

注意事项

此题空间复杂度\(O(n \log 10^9)\)。如果Trie树合并使用新开的结点,每个结构体16B,将需要576MB,这会MLE。考虑Trie树合并时不新开结点,可以将空间降至288MB。

完整代码

 #include<cstdio>
#include<cstring>
#include<vector>
#define DEPTH 30
using namespace std;
struct Trie{
int size;
int next[];
int tag;
}trie[];
int cnt;
int newTrie(){
memset(&trie[++cnt], , sizeof(Trie));
return cnt;
}
inline void pushDown(int i){
int &t = trie[i].tag;
int &l = trie[i].next[], &r = trie[i].next[];
if (t){
if (t & ){ swap(l, r); trie[l].tag++; }
if (t >= ){ trie[l].tag += t / ; trie[r].tag += t / ; }
t = ;
}
}
inline void pushUp(int i){
trie[i].size = trie[trie[i].next[]].size + trie[trie[i].next[]].size;
}
void insert(int i, int depth, int x)
{
if (!depth){ trie[i].size++; return; }
pushDown(i);
int &pos = trie[i].next[x & ];
if (!pos)pos = newTrie();
insert(pos, depth - , x >> );
pushUp(i);
}
void merge(int& i, int j, int k, int depth)
{
if (j&&k){
i = j;
if (!depth){
trie[i].size += trie[k].size;
return;
}
pushDown(j); pushDown(k);
for (int c = ; c < ; c++)
merge(trie[i].next[c], trie[j].next[c], trie[k].next[c], depth - );
pushUp(i);
}
else i = j ? j : k;
}
int f[], id[];
int getFather(int i)
{
if (f[i] == i)return i;
return f[i] = getFather(f[i]);
}
int main()
{
int n, m, x, u, v, k;
scanf("%d%d", &n, &m);
for (int i = ; i <= n; i++){
scanf("%d", &x);
id[i] = newTrie();
f[i] = i;
insert(id[i], DEPTH, x);
}
while (m--){
scanf("%d%d", &x, &u);
u = getFather(u);
if (x == ){
scanf("%d", &v);
v = getFather(v);
if (u != v){
f[u] = v;
merge(id[v], id[u], id[v], DEPTH);
}
}
else if (x == )trie[id[u]].tag++;
else{
scanf("%d%d", &k, &x);
int cur;
for (cur = id[u]; k; k--){
pushDown(cur);
cur = trie[cur].next[x & ];
if (!cur)break;
x >>= ;
}
if (!cur)printf("0\n");
else printf("%d\n", trie[cur].size);
}
}
}

2018 ACM南京网络赛H题Set解题报告的更多相关文章

  1. 2019ICPC南京网络赛A题 The beautiful values of the palace(三维偏序)

    2019ICPC南京网络赛A题 The beautiful values of the palace https://nanti.jisuanke.com/t/41298 Here is a squa ...

  2. 2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告

    2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告 勘误1:第6题第4个 if最后一个条件粗心写错了,答案应为1580. 条件应为abs(a[3]-a[7])!=1,宝宝心理苦啊.!感谢zzh ...

  3. HDU 4758 Walk Through Squares (2013南京网络赛1011题,AC自动机+DP)

    Walk Through Squares Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Oth ...

  4. HDU 4751 Divide Groups (2013南京网络赛1004题,判断二分图)

    Divide Groups Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Tot ...

  5. HDU 4750 Count The Pairs (2013南京网络赛1003题,并查集)

    Count The Pairs Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others ...

  6. ACM-ICPC 2018 焦作赛区网络预赛 H题 String and Times(SAM)

    Now you have a string consists of uppercase letters, two integers AA and BB. We call a substring won ...

  7. 无聊的活动/缘生意转(2018 Nova OJ新年欢乐赛B题)解题报告

    题目2(下面的太抓 我重新写了个背景 其他都一样) 无聊的活动 JLZ老师不情愿的参加了古风社一年一度的活动,他实在不觉得一群学生跳舞有什么好看,更不明白坐在身后的学生为什么这么兴奋(看小姐姐),于是 ...

  8. ACM-ICPC 2018青岛网络赛-H题 Traveling on the Axis

    题目:略(不知道怎么从ZOJ搬题) 地址:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=4054 把这题的每个点分成两种情况 ...

  9. 2018南京网络赛L题:Magical Girl Haze(最短路分层图)

    题目链接:https://nanti.jisuanke.com/t/31001 解题心得: 一个BZOJ的原题,之前就写过博客了. 原题地址:https://www.lydsy.com/JudgeOn ...

随机推荐

  1. 黑幕背后的Autorelease

    http://blog.sunnyxx.com/2014/10/15/behind-autorelease/ 我是前言 Autorelease机制是iOS开发者管理对象内存的好伙伴,MRC中,调用[o ...

  2. GMap.Net解决方案之在WinForm和WPF中使用GMap.Net地图插件的开发

    在做地理位置相关的开发时,总是面临高额地图引擎费用让大部分用户望而却步,加之地图数据又是天价,那么GMap.NET就是首选了,它本身就是开源免费,服务器可以在本地缓存,以后访问时就可以直接访问. 可以 ...

  3. Vue源码学习一 ———— Vue项目目录

    Vue 目录结构 可以在 github 上通过这款 Chrome 插件 octotree 查看Vue的文件目录.也可以克隆到本地.. Vue 是如何规划目录的 scripts ------------ ...

  4. ios swift 里面关于变量 常量 可选类型 控制流的一些心得

    //swift 里面没有头文件和实现文件.只有一个.swift文件 //swift 里面没有main的概念,程序从main.swift开始执行 //swift 每一条执行语句可以不用分号结束,多条语句 ...

  5. MySQL - 表中某个状态字段的状态表示区分最好用数字,如status - [9999:失败,1111:成功]

    表中某个状态字段的状态表示区分最好用数字,如status - [9999:失败,1111:成功]

  6. opengl 学习的链接,以后需要可以再来查需要的

    记录一些好的opengl学习站点,以供日后查阅: modern opengl tutorial : 一个英国的opengl学习网站 上面网站的中文版 日后发现新的再更新

  7. vue 项目中使用mock假数据实现前后端分离

    也是查了很多的资料,整理出来.实现了前后端的分离,用到的技术vue-cli,webpack,node,json-server.首先全局安装json-server cnpm i json-server ...

  8. tp3.2框架中使用volist输出混乱的一点发现

    在tp框架中,volist真的是一样很好用的东西,但是要是不注意,用起来也会有问题的 在Controller层中,将数据assign到页面 $this->assign('vo',$news); ...

  9. RCP 项目启动图片设置

    第一步 选择启动图片命名为 splash.bmp 第二步 添加 扩展点 然 后在右边的扩展元素细节中填入相应的信息,比如我们在这里的application属 性 为 org.vwpolo.rcp.ex ...

  10. OpenCV学习笔记(三) 访问像素

    转自:OpenCV如何扫描图像.利用查找表和计时 测试代码:opencv\samples\cpp\tutorial_code\core\how_to_scan_images 测试函数耗时 cv::ge ...