题目描述

n个集合 m个操作
操作:
1 a b 合并a,b所在集合
2 k 回到第k次操作之后的状态(查询算作操作)
3 a b 询问a,b是否属于同一集合,是则输出1否则输出0

0

输入格式

输出格式

输入样例

5 6

1 1 2

3 1 2

2 0

3 1 2

2 1

3 1 2

输出样例

1

0

1

题解

这道题要维护可持续化并查集,由于并查集是由数组实现的,所以实质是维护并查集的pre数组
路径压缩怎么办?实际上可以按轶合并,轶指最深的深度
每次合并集合时,将轶小的并到轶大的,当二者相等,被并的轶+1,即最大深度+1
这样子维护的并查集近似于完全二叉树,可以做到查询均摊O(logn)

由于没怎么写过可持续化数组,这里讲一讲:
可持续化数组,实际上就是可持续化线段树。可以看做废掉了中间节点的主席树,每次修改和查询都一样,无论是空间还是时间都是O(logn)

我们先开一个0版本线段树,每个叶子节点有一个值,表示对应位置的数组的值

每次修改,加一个版本的根,然后让新版本的树沿着上一版本创建。有修改的那一条路径新开节点,剩余的子树指向原版本【因为本来就一样】

每次询问,只需找到对应版本的根,往叶子查找即可

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define LL long long int
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define Redge(u) for (int k = h[u]; k != -1; k = ed[k].nxt)
using namespace std;
const int maxn = 20005,maxm = 2000005,INF = 1000000000;
inline int RD(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57) {out = (out << 1) + (out << 3) + c - '0'; c = getchar();}
return out * flag;
}
int N,M,siz = 0,rt[maxn],ri = 0;
struct node{int ls,rs,fa,dep;}e[maxm];
void build(int& u,int l,int r){
u = ++siz;
if (l == r){e[u].fa = l; return;}
int mid = l + r >> 1;
build(e[u].ls,l,mid);
build(e[u].rs,mid + 1,r);
}
int Query(int u,int l,int r,int pos){
if (l == r) return u;
int mid = l + r >> 1;
if (mid >= pos) return Query(e[u].ls,l,mid,pos);
else return Query(e[u].rs,mid + 1,r,pos);
}
void modify(int& u,int pre,int l,int r,int pos,int val){
e[u = ++siz] = e[pre];
if (l == r) {e[u].fa = val; return;}
int mid = l + r >> 1;
if (mid >= pos) modify(e[u].ls,e[pre].ls,l,mid,pos,val);
else modify(e[u].rs,e[pre].rs,mid + 1,r,pos,val);
}
void add(int u,int l,int r,int pos){
if (l == r) {e[u].dep++; return;}
int mid = l + r >> 1;
if (mid >= pos) add(e[u].ls,l,mid,pos);
else add(e[u].rs,mid + 1,r,pos);
}
int find(int R,int u){
int p = Query(R,1,N,u);
if (e[p].fa == u) return p;
return find(R,e[p].fa);
}
int main(){
N = RD(); M = RD(); int cmd,a,b,fa,fb;
build(rt[0],1,N);
REP(i,M){
cmd = RD(); a = RD(); ri++;
if (cmd == 1){
b = RD(); rt[i] = rt[i - 1];
fa = find(rt[i],a); fb = find(rt[i],b);
if (e[fa].fa != e[fb].fa){
if (e[fa].dep > e[fb].dep) swap(fa,fb);
modify(rt[ri],rt[ri - 1],1,N,e[fa].fa,e[fb].fa);
if (e[fa].dep == e[fb].dep) add(rt[ri],1,N,e[fb].fa);
}
}else if (cmd == 2){
rt[ri] = rt[a];
}else {
b = RD(); rt[ri] = rt[ri - 1];
fa = find(rt[ri],a); fb = find(rt[ri],b);
printf("%d\n",fa == fb);
}
}
return 0;
}

题目描述

Description:
自从zkysb出了可持久化并查集后……
hzwer:乱写能AC,暴力踩标程
KuribohG:我不路径压缩就过了!
ndsf:暴力就可以轻松虐!
zky:……

n个集合 m个操作
操作:
1 a b 合并a,b所在集合
2 k 回到第k次操作之后的状态(查询算作操作)
3 a b 询问a,b是否属于同一集合,是则输出1否则输出0
请注意本题采用强制在线,所给的a,b,k均经过加密,加密方法为x = x xor lastans,lastans的初始值为0
0

输入格式

输出格式

输入样例

5 6

1 1 2

3 1 2

2 1

3 0 3

2 1

3 1 2

输出样例

1

0

1

题解

实际是一样的,O(nlog2n)的复杂度怎么卡得掉

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define LL long long int
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define Redge(u) for (int k = h[u]; k != -1; k = ed[k].nxt)
using namespace std;
const int maxn = 200005,maxm = 10000005,INF = 1000000000;
inline int RD(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57) {out = (out << 1) + (out << 3) + c - '0'; c = getchar();}
return out * flag;
}
int N,M,siz = 0,rt[maxn],ri = 0;
struct node{int ls,rs,v,dep;}e[maxm];
void build(int& u,int l,int r){
if (!u) u = ++siz;
if (l == r){e[u].v = l; return;}
int mid = l + r >> 1;
build(e[u].ls,l,mid);
build(e[u].rs,mid + 1,r);
}
int Query(int u,int l,int r,int pos){
if (l == r) return u;
int mid = l + r >> 1;
if (mid >= pos) return Query(e[u].ls,l,mid,pos);
else return Query(e[u].rs,mid + 1,r,pos);
}
void modify(int& u,int pre,int l,int r,int pos,int val){
u = ++siz;
if (l == r) {e[u].v = val; e[u].dep = e[pre].dep; return;}
e[u].ls = e[pre].ls; e[u].rs = e[pre].rs;
int mid = l + r >> 1;
if (mid >= pos) modify(e[u].ls,e[pre].ls,l,mid,pos,val);
else modify(e[u].rs,e[pre].rs,mid + 1,r,pos,val);
}
void add(int u,int l,int r,int pos){
if (l == r) {e[u].dep++; return;}
int mid = l + r >> 1;
if (mid >= pos) add(e[u].ls,l,mid,pos);
else add(e[u].rs,mid + 1,r,pos);
}
int find(int R,int u){
int p = Query(R,1,N,u);
if (e[p].v == u) return p;
return find(R,e[p].v);
}
int main(){
N = RD(); M = RD(); int cmd,a,b,p,q,last = 0;
build(rt[0],1,N);
REP(i,M){
cmd = RD(); a = RD() ^ last; ri++;
if (cmd == 1){
b = RD() ^ last; rt[i] = rt[i - 1];
p = find(rt[i],a); q = find(rt[i],b);
if (e[p].v != e[q].v){
if (e[p].dep > e[q].dep) swap(p,q);
modify(rt[ri],rt[ri - 1],1,N,e[p].v,e[q].v);
if (e[p].dep == e[q].dep) add(rt[ri],1,N,e[q].v);
}
}else if (cmd == 2){
rt[ri] = rt[a];
}else {
b = RD() ^ last; rt[ri] = rt[ri - 1];
p = find(rt[ri],a); q = find(rt[ri],b);
if (e[p].v == e[q].v) last = 1;
else last = 0;
printf("%d\n",last);
}
}
return 0;
}

最新文章

  1. 自动执行任务管理---TaskManage
  2. dict
  3. 人脸识别经典算法三:Fisherface(LDA)
  4. Nginx系列2之Nginx+php
  5. css控制页面打印(分页、屏蔽不需要打印的对象)
  6. MyEclipse------如何添加jspsmartupload.jar+文件上传到服务器
  7. Linux命令之exit - 退出当前shell【返回值状态】
  8. Android 4.0源码目录结构
  9. mysql中union与union all的区别
  10. wpf 调用线程必须为sta 因为许多ui组件都需要
  11. jquery获取checkbox被选中的值
  12. JAVA必备——13个核心规范
  13. java下发电子邮件demo
  14. Unity用Excel.dll简单读取Excel内容
  15. 0.计划用libgdx写一个六边形回合制slg兵棋游戏
  16. Parameter infoDTOs of type T from private T com.ListVO.setInfoDTOs is not resolvable to a concrete type.
  17. 《Linux内核分析》 第三周 构造一个简单的Linux系统MenuOS
  18. loj2048 「HNOI2016」最小公倍数
  19. .net core程序部署
  20. android 签名验证防止重打包

热门文章

  1. 解决url传递过程中加号变空格的问题
  2. 在一台Apache服务器上创建多个站点(不同域名)
  3. python函数的返回值
  4. ecshop 漏洞如何修复 补丁升级与安全修复详情
  5. 【Python让生活更美好01】os与shutil模块的常用方法总结
  6. The Road to learn React书籍学习笔记(第四章)
  7. c/c++ 数组传参
  8. 触发显示和隐藏 div
  9. n个台阶,每次都可以走一步,走两步,走三步,走到顶部一共有多少种可能
  10. Qt 汽车仪表再次编写,Widget,仪表显示,绘制界面