题目描述

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;
}

BZOJ3673 & BZOJ3674 可持续化并查集 【可持续化线段树维护可持续化数组】的更多相关文章

  1. 【BZOJ2054】疯狂的馒头(并查集,线段树)

    [BZOJ2054]疯狂的馒头(并查集,线段树) 题面 BZOJ 然而权限题,随便找个离线题库看看题吧. 题解 线段树就是个暴力,如果数据可以构造就能卡掉,然而不能构造,要不然复杂度瓶颈成为了读入了. ...

  2. UVALive - 5031 Graph and Queries (并查集+平衡树/线段树)

    给定一个图,支持三种操作: 1.删除一条边 2.查询与x结点相连的第k大的结点 3.修改x结点的权值 解法:离线倒序操作,平衡树or线段树维护连通块中的所有结点信息,加个合并操作就行了. 感觉线段树要 ...

  3. POJ 1944 Fiber Communications (枚举 + 并查集 OR 线段树)

    题意 在一个有N(1 ≤ N ≤ 1,000)个点环形图上有P(1 ≤ P ≤ 10,000)对点需要连接.连接只能连接环上相邻的点.问至少需要连接几条边. 思路 突破点在于最后的结果一定不是一个环! ...

  4. ACM学习历程—SNNUOJ 1110 传输网络((并查集 && 离线) || (线段树 && 时间戳))(2015陕西省大学生程序设计竞赛D题)

    Description Byteland国家的网络单向传输系统可以被看成是以首都 Bytetown为中心的有向树,一开始只有Bytetown建有基站,所有其他城市的信号都是从Bytetown传输过来的 ...

  5. 2019牛客第八场多校 E_Explorer 可撤销并查集(栈)+线段树

    目录 题意: 分析: @(2019牛客暑期多校训练营(第八场)E_Explorer) 题意: 链接 题目类似:CF366D,Gym101652T 本题给你\(n(100000)\)个点\(m(1000 ...

  6. [bzoj3673][可持久化并查集 by zky] (rope(可持久化数组)+并查集=可持久化并查集)

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

  7. HDU 3974 Assign the task 并查集/图论/线段树

    Assign the task Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?p ...

  8. bzoj3673可持久化并查集 by zky&&bzoj3674可持久化并查集加强版

    bzoj3673可持久化并查集 by zky 题意: 维护可以恢复到第k次操作后的并查集. 题解: 用可持久化线段树维护并查集的fa数组和秩(在并查集里的深度),不能路径压缩所以用按秩启发式合并,可以 ...

  9. bzoj3673可持久化并查集

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

随机推荐

  1. MySQL innodb表使用表空间物理文件复制或迁移表

    MySQL InnoDB引擎的表通过拷贝物理文件来进行单表或指定表的复制,可以想到多种方式,今天测试其中2种: 将innodb引擎的表修改为Myisam引擎,然后拷贝物理文件 直接拷贝innodb的表 ...

  2. Java : 多态表现:静态绑定与动态绑定(向上转型的运行机制)

    本来想自己写写的,但是看到有人分析的可以说是很清晰了,故转过来. 原文地址:http://www.cnblogs.com/ygj0930/p/6554103.html 一:绑定 把一个方法与其所在的类 ...

  3. json_decode结果为null的几种原因

    值只能是UTF-8编码,元素最后不能有逗号,元素不能使用单引号,元素值中间不能有空格和n.

  4. python入门——Anaconda安装

    初学Python,可以选择python原始的IDE,但原始的IDE在使用过程中需要自己安装各种包,个人觉得初学者不需要将时间花在这些上面,而是应该直接学习python程序,这些比较杂的事情可以在以后的 ...

  5. 【APUE】Chapter16 Network IPC: Sockets & makefile写法学习

    16.1 Introduction Chapter15讲的是同一个machine之间不同进程的通信,这一章内容是不同machine之间通过network通信,切入点是socket. 16.2 Sock ...

  6. Ubuntu 手机 app开发学习0

    # 相关网址 http://developer.ubuntu.com/zh-cn/apps/sdk/ 0. 环境搭建 首选需要一个Ubuntu 14.04操作系统.没啥好讲的,直接安装了一个虚拟机. ...

  7. Mootools 学习随笔

    简单的介绍下Mootools: MooTools是一个简洁,模块化,面向对象的开源JavaScript web应用框架.在处理js.css.html时候,为web开发者提供了一个跨浏览器的js解决方案 ...

  8. HDFS分布式集群

    一.HDFS伪分布式环境搭建 Hadoop分布式文件系统(HDFS)被设计成适合运行在通用硬件(commodity hardware)上的分布式文件系统.它和现有的分布式文件系统有很多共同点.但同时, ...

  9. 了解游戏编程与 AI

    噫语系列... 闲话 最近在重写我的一个 QQ 群机器人项目,并尝试将它改成更通用的结构,以方便在未来加入对 Wechat 和 Telegram 的支持. 在查资料的过程中,发现很多人认为一个群内多人 ...

  10. MyBatis实例教程--开发环境搭建

    MyBatis实例教程--开发环境搭建 准备工作: 1.mybatis 的开发环境搭建,选择: eclipse j2ee 版本,mysql 5.1 ,jdk 1.7,mybatis3.2.0.jar包 ...