n个集合 m个操作

操作:

  • 1 a b 合并a,b所在集合

  • 2 k 回到第k次操作之后的状态(查询算作操作)

  • 3 a b 询问a,b是否属于同一集合,是则输出1否则输出0

说是可持久化并查集,实际上是把并查集的所有find和merge操作都放到可持久化数组上做,这样可以做到完全可持久化(不仅能查询某个历史版本,而且还能在某个历史版本的基础上进行修改)

注意并查集要按秩合并

可持久化数组可以用可持久化线段树(主席树)或者可持久化平衡树实现,理论上平衡树应该更快一些而且更省内存(但也省不了多少)

线段树版:

 #include<bits/stdc++.h>
using namespace std;
const int N=2e5+;
int rt[N],fa[N*],mxd[N*],ls[N*],rs[N*],tot,n,m;
#define mid ((l+r)>>1)
int cpy(int v) {int u=++tot; fa[u]=fa[v],mxd[u]=mxd[v],ls[u]=ls[v],rs[u]=rs[v]; return u;}
int qry(int* a,int p,int& u,int l=,int r=n) {
if(l==r)return a[u];
return p<=mid?qry(a,p,ls[u],l,mid):qry(a,p,rs[u],mid+,r);
}
void upd(int* a,int p,int x,int v,int& u,int l=,int r=n) {
u=cpy(v);
if(l==r) {a[u]=x; return;}
p<=mid?upd(a,p,x,ls[v],ls[u],l,mid):upd(a,p,x,rs[v],rs[u],mid+,r);
}
int fd(int x,int u) {
int fx=qry(fa,x,u);
return fx?fd(fx,u):x;
}
void mg(int x,int y,int& u) {
int fx=fd(x,u),fy=fd(y,u);
if(fx==fy)return;
int dx=qry(mxd,fx,u),dy=qry(mxd,fy,u);
if(dx>dy)swap(fx,fy),swap(dx,dy);
upd(fa,fx,fy,u,u);
if(dx==dy)upd(mxd,fy,dx+,u,u);
}
int main() {
mxd[]=;
scanf("%d%d",&n,&m);
for(int i=; i<=m; ++i) {
rt[i]=rt[i-];
int f,x,y;
scanf("%d",&f);
if(f==)scanf("%d%d",&x,&y),mg(x,y,rt[i]);
else if(f==)scanf("%d",&x),rt[i]=rt[x];
else scanf("%d%d",&x,&y),printf("%d\n",fd(x,rt[i])==fd(y,rt[i]));
}
return ;
}

平衡树版:(由于没有旋转分裂等操作所以代码和线段树基本没区别,只是二分判断的条件稍微改了一下)

 #include<bits/stdc++.h>
using namespace std;
const int N=2e5+;
int rt[N],fa[N*],mxd[N*],ls[N*],rs[N*],tot,n,m;
#define mid ((l+r)>>1)
int cpy(int v) {int u=++tot; fa[u]=fa[v],mxd[u]=mxd[v],ls[u]=ls[v],rs[u]=rs[v]; return u;}
int qry(int* a,int p,int& u,int l=,int r=n) {
if(p==mid)return a[u];
return p<mid?qry(a,p,ls[u],l,mid-):qry(a,p,rs[u],mid+,r);
}
void upd(int* a,int p,int x,int v,int& u,int l=,int r=n) {
u=cpy(v);
if(p==mid) {a[u]=x; return;}
p<mid?upd(a,p,x,ls[v],ls[u],l,mid-):upd(a,p,x,rs[v],rs[u],mid+,r);
}
int fd(int x,int u) {
int fx=qry(fa,x,u);
return fx?fd(fx,u):x;
}
void mg(int x,int y,int& u) {
int fx=fd(x,u),fy=fd(y,u);
if(fx==fy)return;
int dx=qry(mxd,fx,u),dy=qry(mxd,fy,u);
if(dx>dy)swap(fx,fy),swap(dx,dy);
upd(fa,fx,fy,u,u);
if(dx==dy)upd(mxd,fy,dx+,u,u);
}
int main() {
mxd[]=;
scanf("%d%d",&n,&m);
for(int i=; i<=m; ++i) {
rt[i]=rt[i-];
int f,x,y;
scanf("%d",&f);
if(f==)scanf("%d%d",&x,&y),mg(x,y,rt[i]);
else if(f==)scanf("%d",&x),rt[i]=rt[x];
else scanf("%d%d",&x,&y),printf("%d\n",fd(x,rt[i])==fd(y,rt[i]));
}
return ;
}

洛谷P3402 可持久化并查集的更多相关文章

  1. bzoj3673 & bzoj3674 & 洛谷P3402 可持久化并查集

    题目:bzoj3673:https://www.lydsy.com/JudgeOnline/problem.php?id=3673 bzoj3674:https://www.lydsy.com/Jud ...

  2. P3402 可持久化并查集

    P3402 通过主席树维护不同版本的并查集,注意要采用按秩合并的方式,路径压缩可能会爆. 1 #include <bits/stdc++.h> 2 using namespace std; ...

  3. 洛谷 3295 [SCOI2016]萌萌哒——并查集优化连边

    题目:https://www.luogu.org/problemnew/show/P3295 当要连的边形如 “一段区间内都是 i 向 i+L 连边” 的时候,用并查集优化连边. 在连边的时候,如果要 ...

  4. 洛谷P2024 食物链 [NOI2001] 并查集

    正解:并查集 解题报告: 传送门(咕了! 其实没有很难(虽然我是交了三发才过的QAQ 但是一来好久没打并查集了恢复一下智力 二来看着智推里唯一一个蓝就很不爽(,,,虽然做了这题之后又补上了个蓝题QAQ ...

  5. 洛谷P1197 [JSOI2008] 星球大战 [并查集]

    题目传送门 星球大战 题目描述 很久以前,在一个遥远的星系,一个黑暗的帝国靠着它的超级武器统治者整个星系. 某一天,凭着一个偶然的机遇,一支反抗军摧毁了帝国的超级武器,并攻下了星系中几乎所有的星球.这 ...

  6. 洛谷 P1551 亲戚(并查集模板)

    嗯... 题目链接:https://www.luogu.org/problemnew/show/P1551 思路: 很显然地我们会发现,这是一道并查集的模板题,并且是考察了并查集中的”并“和”查“的操 ...

  7. 洛谷P1111修复公路并查集改

    看了他们的题解感觉很震惊,为什么要用kruskal,这题要用到最小生成树吗??? 38行短短的程序就可以了,我觉得学习不是一种套用,套自己学的,而且题解很大一部分都是kruskal. 个人认为自己的程 ...

  8. 洛谷P1525关押罪犯——并查集

    题目:https://www.luogu.org/problemnew/show/P1525 并查集+贪心,从大到小排序,将二人分在不同房间,找到第一个不满足的即为答案. 代码如下: #include ...

  9. 洛谷 - P5429 - Fence Planning - 并查集

    https://www.luogu.org/problemnew/show/P5429 很明显是要维护整个连通块的共同性质,并查集一搞就完事了. #include<bits/stdc++.h&g ...

随机推荐

  1. 掌握Pod-Pod调度策略

    一 Pod生命周期管理 1.1 Pod生命周期 Pod在整个生命周期过程中被系统定义了如下各种状态. 状态值 描述 Pending API Server已经创建该Pod,且Pod内还有一个或多个容器的 ...

  2. Sqlserver实现故障转移 — AlwaysOn实现故障转移(4)

    目的:在已经加域的计算机上安装sqlserver2012,并配置系统级故障转移及数据库,实现AlwayOn. 域控的建立详见:https://www.cnblogs.com/xiaoerlang90/ ...

  3. C# 线程thread

    一.问题总结 1. 在WinForm开发过程中用到线程时,往往需要在线程中访问线程外的控件,比如:设置textbox的Text值等等.如果直接访问UI控件会报出“从不是创建控件的线程访问它”错误.控件 ...

  4. Java并发编程之程序运行堆栈分析

    Java程序运行的堆栈分析 1.JVM运行时数据区 JVM通过加载class文件的数据来执行程序.JVM在运行时会划分不同的区域以存放数据.如下图所示: 线程共享部分:所有线程都能访问这块内存的数据, ...

  5. 基于OpenCV的三维数据点的曲面重构_MySurefaceReconstruction

    在Opencv中有个Viz模块,可以显示三维物体,还可以实现三维动画,本来是很好的东东,但是里面的函数.类的说明太过简单,始终不得要领.不过其中一个扩展功能非常好,就是你可以在vtk中设计自己的模型类 ...

  6. vue中refs的使用

    最近在看其他项目的过程中,发现在dom节点上使用了ref="xxx"的使用,以前一直不知道该属性起着什么作用,因为一直忙着写项目. 这两天项目不忙了,有闲心来看别人做的项目了,就看 ...

  7. Star all over again.

    0x00前言 经过了一上午的折腾之后,博客的界面勉强可观,今天下午将之前的所有博客全部删除,重新开始写属于自己的博客,而不是只把它当作一个收藏夹,转载其他人的文章. 0x01近来感想 有感而发,随便写 ...

  8. 【C/C++】对于可重入、线程安全、异步信号安全几个概念的理解

    由于前段时间,程序偶尔异常挂起不工作,检查后发现时死锁了,原因就是:在信号处理函数里面调用了fprintf. printf等io函数是需要对输出缓冲区加锁,这类函数对本身是线程安全的,但是对信号处理函 ...

  9. 【Python开发】Python:itertools模块

    Python:itertools模块 itertools模块包含创建有效迭代器的函数,可以用各种方式对数据进行循环操作,此模块中的所有函数返回的迭代器都可以与for循环语句以及其他包含迭代器(如生成器 ...

  10. Flume概述

    flume是分布式的,可靠的,用于从不同的来源有效收集 聚集 和 移动 大量的日志数据用以集中式的数据存储的系统. 是apache的一个顶级项目. 系统需求:jdk1.6以上,推荐java1.7