可持久化并查集小结

https://www.zybuluo.com/ysner/note/1253722

定义

允许恢复历史状态的并查集。

建立

\(Q\)棵主席树,每个主席树上维护当前状态并查集各个节点的父亲。
(实际上就是并查集和主席树强行捆绑在一起)

操作

每次操作前自动继承上次操作后的状态。

  • 合并\(a,b\)所在集合
    把两棵主席树按秩合并(深度大的合并到深度小的)。
    如果两棵主席树合并时深度相等,给合并后的主席树深度\(+1\)(要不然哪来的秩)

  • 回到第\(k\)次操作之后的状态
    把当前主席树的根赋值为第\(k\)棵主席树的即可。

  • 询问\(a\),\(b\)是否属于同一集合
    并查集(主)+主席树(辅)查询两者祖先,比较是否相等。

用途

  • \(bzoj\)上刷\(AC\)
  • ???
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<cstdlib>
#include<algorithm>
#define ll long long
#define re register
#define il inline
#define ls t[x][0]
#define rs t[x][1]
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
#define fp(i,a,b) for(re int i=a;i<=b;i++)
#define fq(i,a,b) for(re int i=a;i>=b;i--)
using namespace std;
const int N=5e6+100;
int n,m,rt[N],tot,f[N],t[N][2],d[N];
il int gi()
{
  re int x=0,t=1;
  re char ch=getchar();
  while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
  if(ch=='-') t=-1,ch=getchar();
  while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar();
  return x*t;
}
il void Build(re int &x,re int l,re int r)
{
  x=++tot;
  if(l==r) {f[x]=l;return;}
  re int mid=l+r>>1;
  Build(ls,l,mid);Build(rs,mid+1,r);
}
il void Modify(re int &x,re int las,re int l,re int r,re int pos,re int ff)
{
  x=++tot;
  if(l==r) {f[x]=ff;d[x]=d[las];return;}
  re int mid=l+r>>1;
  ls=t[las][0];rs=t[las][1];
  if(pos<=mid) Modify(ls,t[las][0],l,mid,pos,ff);
  else Modify(rs,t[las][1],mid+1,r,pos,ff);
}
il int Query(re int x,re int l,re int r,re int pos)
{
  if(l==r) return x;
  re int mid=l+r>>1;
  if(pos<=mid) return Query(ls,l,mid,pos);
  else return Query(rs,mid+1,r,pos);
}
il void add(re int x,re int l,re int r,re int pos)
{
  if(l==r) {++d[x];return;}
  re int mid=l+r>>1;
  if(pos<=mid) add(ls,l,mid,pos);
  else add(rs,mid+1,r,pos);
}
il int find(re int rt,re int x)
{
  re int fa=Query(rt,1,n,x);
  return x==f[fa]?fa:find(rt,f[fa]);
}
int main()
{
  n=gi();m=gi();
  Build(rt[0],1,n);
  fp(i,1,m)
    {
      re int op=gi();
      if(op==1)
    {
      rt[i]=rt[i-1];
      re int x=find(rt[i],gi()),y=find(rt[i],gi());
      if(f[x]==f[y]) continue;
      if(d[x]>d[y]) swap(x,y);
      Modify(rt[i],rt[i-1],1,n,f[x],f[y]);
      if(d[x]==d[y]) add(rt[i],1,n,f[y]);
    }
      if(op==2) rt[i]=rt[gi()];
      if(op==3)
    {
      rt[i]=rt[i-1];
      re int x=find(rt[i],gi()),y=find(rt[i],gi());
      puts(f[x]==f[y]?"1":"0");
    }
    }
  return 0;
}
posted @ 2018-08-17 22:12  小蒟蒻ysn  阅读(192)  评论(0编辑  收藏  举报