zoukankan      html  css  js  c++  java
  • BZOJ3673 可持久化并查集 by zky <可持久化数组+主席树>

    可持久化并查集 by zky

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

    Sample Input
    5 6
    1 1 2
    3 1 2
    2 0
    3 1 2
    2 1
    3 1 2
    Sample Output
    1
    0
    1
    Hint
    0 < n,m <= 2*10^4

    标签:主席树维护可持久化数组

    本题的做法其实和并查集没太大关联。
    如果是撤销,那可以用不加路径压缩的并查集完成,但是如果回到某时间,则不太好写。
    这时,我们发现并查集这个东西构造其实很简单,只需要一个fa数组就行了。所以我们自然可以想到直接用一个二维数组存储每个时间的fa数组,即用增加的一维表示时间。
    但是,n、m是2e4级别,所以肯定会MLE,这里我们就需要用到主席树,把fa数组可持久化。这里需要注意我们不能用路径压缩优化,因为我们需要回到前面的状态,为了让它跑得更快,我们可以用按秩合并优化。
    最后附上AC代码:

    #include <iostream>
    #include <cstdio>
    #define MAX_N 20000
    using namespace std;
    int n, m, cnt, now, root[MAX_N*10+5];
    struct node {int fa, dep, ls, rs;} tr[MAX_N*100+5];
    void build(int v, int l, int r) {
    	if (l == r) {tr[v].fa = l;	return;}
    	int mid = l+r>>1;
    	tr[v].ls = ++cnt, tr[v].rs = ++cnt;
    	build(tr[v].ls, l, mid);
    	build(tr[v].rs, mid+1, r);
    }
    void modifyfa(int v, int o, int s, int t, int pos, int x) {
    	tr[v] = tr[o];
    	if (s == t)	{tr[v].fa = x;	return;}
    	int mid = s+t>>1;
    	if (pos <= mid)	modifyfa(tr[v].ls = ++cnt, tr[o].ls, s, mid, pos, x);
    	else	modifyfa(tr[v].rs = ++cnt, tr[o].rs, mid+1, t, pos, x);
    }
    void modifydep(int v, int s, int t, int pos) {
    	if (s == t)	{tr[v].dep++;	return;}
    	int mid = s+t>>1;
    	if (pos <= mid)	modifydep(tr[v].ls, s, mid, pos);
    	else	modifydep(tr[v].rs, mid+1, t, pos);
    }
    int find(int v, int s, int t, int pos) {
    	if (s == t)	return v;
    	int mid = s+t>>1;
    	if (pos <= mid)	return find(tr[v].ls, s, mid, pos);
    	else	return find(tr[v].rs, mid+1, t, pos);
    }
    int getf(int r, int x) {
    	int pos = find(r, 1, n, x);
    	if (tr[pos].fa != x)	return getf(r, tr[pos].fa);
    	return pos;
    }
    int main() {
    	scanf("%d%d", &n, &m);
    	cnt = 0, root[0] = ++cnt;
    	build(root[0], 1, n);
    	for (int now = 1; now <= m; now++) {
    		int opt;
    		scanf("%d", &opt);
    		if (opt == 1) {
    			root[now] = root[now-1];
    			int a, b;
    			scanf("%d%d", &a, &b);
    			int posa = getf(root[now], a), posb = getf(root[now], b);
    			if (tr[posa].fa == tr[posb].fa)	continue;
    			if (tr[posa].dep > tr[posb].dep)	swap(posa, posb);
    			root[now] = ++cnt;
    			modifyfa(root[now], root[now-1], 1, n, tr[posa].fa, tr[posb].fa);
    			if (tr[posa].dep == tr[posb].dep)	modifydep(root[now], 1, n, tr[posb].fa);
    		}
    		if (opt == 2) {
    			int k;
    			scanf("%d", &k);
    			root[now] = root[k];
    		}
    		if (opt == 3) {
    			root[now] = root[now-1];
    			int a, b;
    			scanf("%d%d", &a, &b);
    			if (tr[getf(root[now], a)].fa == tr[getf(root[now], b)].fa)	printf("1
    ");
    			else	printf("0
    ");
    		}
    	}
    	return 0;
    }
    
  • 相关阅读:
    设置IIS7/IIS7.5的FTP支持断点续传
    win10图片打开方式里没有默认照片查看器的解决方法
    使用C#的HttpWebRequest模拟登陆网站
    在Raid模式下装Win10找不到固态硬盘怎么办
    C# String 前面不足位数补零的方法
    [SQL Server]无法创建 SSIS 运行时对象,请验证 DTS.dll 是否可用及是否已注册
    HTTPS协议说明
    技术晨读_20160217
    技术晨读_20160215
    fastcgi安装
  • 原文地址:https://www.cnblogs.com/AzraelDeath/p/7561726.html
Copyright © 2011-2022 走看看