首页 > 代码库 > [bzoj3673][可持久化并查集 by zky] (rope(可持久化数组)+并查集=可持久化并查集)
[bzoj3673][可持久化并查集 by zky] (rope(可持久化数组)+并查集=可持久化并查集)
Description
n个集合 m个操作
操作:
1 a b 合并a,b所在集合
2 k 回到第k次操作之后的状态(查询算作操作)
3 a b 询问a,b是否属于同一集合,是则输出1否则输出0
0<n,m<=2*10^4
Input
Output
Sample Input
5 61 1 23 1 22 03 1 22 13 1 2
Sample Output
101
Solution
用rope实现可持久化数组,用rope的历史记录功能实现可持久化并查集,通过时间168ms
#include<cstdio>#include<ext/rope>#include<iostream>using namespace std;using namespace __gnu_cxx;inline int Rin(){ int x=0,c=getchar(),f=1; for(;c<48||c>57;c=getchar()) if(!(c^45))f=-1; for(;c>47&&c<58;c=getchar()) x=(x<<1)+(x<<3)+c-48; return x*f;}const int M=20010;rope<int>*his[M];int n,m,a[M];inline int find(int i,int x){ int d=x,p; while(his[i]->at(x)^x)x=his[i]->at(x); while(d^x)p=his[i]->at(d),his[i]->replace(d,x),d=p; return x;}inline void merge(int i,int x,int y){ x=find(i,x),y=find(i,y); if(x^y) his[i]->replace(y,x);}int main(){ n=Rin(),m=Rin(); for(int i=1;i<=n;i++)a[i]=i; his[0]=new rope<int>(a,a+1+n); for(int i=1,x,y,sign;i<=m;i++){ his[i]=new rope<int>(*his[i-1]); scanf("%d",&sign); switch(sign){ case 1: x=Rin(),y=Rin(); merge(i,x,y); break; case 2: x=Rin(); his[i]=his[x]; break; case 3: x=Rin(),y=Rin(); printf("%d\n",(!(find(i,x)^find(i,y)))); break; } } return 0;}
[bzoj3673][可持久化并查集 by zky] (rope(可持久化数组)+并查集=可持久化并查集)
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。