zoukankan      html  css  js  c++  java
  • 【洛谷 P1653】 猴子 (并查集)

    题目链接
    没删除调试输出,原地炸裂,(80)->(0)。如果你要问剩下的(20)呢?答:数组开小了。
    这题正向删边判连通性是很不好做的,因为我们并不会并查集的逆操作。于是可以考虑把断边改成逆向连边,某个猴子什么时候和(1)号猴子变成连通的,这就是他掉下去的时间,如果本来就与(1)号猴子连通,那么它就永远不会掉下去。我用了两个并查集,一个维护连通性,一个维护答案,这两个并查集前面的操作几乎是一模一样的,同时连边,到最后一步也就是某个猴子与(1)号猴子变为连通时,前者正常连边,后者不连,因为我们要记录原连通块里的猴子答案是一样的。然后就(ok)了。

    #include <cstdio>
    #include <cstring>
    #define Open(s) freopen(s".in","r",stdin);freopen(s".out","w",stdout);
    #define Close fclose(stdin);fclose(stdout);
    const int MAXN = 500010;
    inline int read(){
    	int s = 0, w = 1;
    	char ch = getchar();
    	while(ch < '0' || ch > '9'){ if(ch == '-') w = -1; ch = getchar();}
    	while(ch >= '0' && ch <= '9') { s = s * 10 + ch - '0'; ch = getchar(); }
    	return s * w;
    }
    int n, m;
    int a[MAXN], b[MAXN], f[MAXN], ban[MAXN][3], F[MAXN];
    int fall_time[MAXN], son[MAXN][3], start_set[MAXN], linked[MAXN];
    int junk;
    void init(){
    	for(int i = 1; i <= n; ++i)
    		f[i] = i;
    }
    int find(int x){
    	return f[x] == x ? x : f[x] = find(f[x]);
    }
    void merge(int x, int y){
    	f[find(y)] = find(x);
    }
    void Init(){
    	for(int i = 1; i <= n; ++i)
    		F[i] = i;
    }
    int Find(int x){
    	return F[x] == x ? x : F[x] = Find(F[x]);
    }
    void Merge(int x, int y){
    	F[Find(y)] = Find(x);
    }
    int main(){
    	Open("monkey");
    	memset(fall_time, -1, sizeof fall_time);
    	n = read(); m = read();
    	init(); Init();
    	for(int i = 1; i <= n; ++i){
    		son[i][1] = read(); son[i][2] = read();
    	}
    	for(int i = 1; i <= m; ++i){
    		a[i] = read(); b[i] = read();
    		ban[a[i]][b[i]] = 1;
    	}
    	for(int i = 1; i <= n; ++i)
    		for(int j = 1; j <= 2; ++j)
    			if(!ban[i][j])
    				if(son[i][j] != -1)
    					merge(i, son[i][j]), Merge(i, son[i][j]);
    	for(int i = m; i; --i){
    		junk = son[a[i]][b[i]];
    		int fa = find(a[i]) != find(1), fb = find(junk) != find(1);
    		merge(a[i], junk);
    		if(fa && find(a[i]) == find(1))
    			fall_time[Find(a[i])] = i - 1;
    		else if(fa) Merge(a[i], junk);
    		if(fb && find(junk) == find(1))
    			fall_time[Find(junk)] = i - 1;
    		else if(fb) Merge(a[i], junk);
    	}
    	for(int i = 1; i <= n; ++i)
    		printf("%d
    ", fall_time[Find(i)]);
    	Close;
    	return 0;
    }
    
    
  • 相关阅读:
    hdu4651(广义五边形数 & 分割函数1)
    Java基础面试题1
    Java8新特性--Lambda表达式
    Java中list在循环中删除元素的坑
    Java多线程面试题整理
    Java并发包--ConcurrentHashMap原理解析
    HashMap原理解析
    Java原子类--AtomicLongFieldUpdater
    Java原子类--AtomicReference
    Java原子类--AtomicLongArray
  • 原文地址:https://www.cnblogs.com/Qihoo360/p/9660259.html
Copyright © 2011-2022 走看看