题目大意:
城市里有两个帮派,D a b 代表ab不是一个帮派的,A a b是问你这两个人事不是同一帮派的;
考察并查集
有两种算法;
第一种算法:D a b 的时候将a 和b+n 放到一个并查集,a+n和b放到一个并查集里;
我们就会得到如下的一个图;
1 1+n
2 2+n
3 3+n
。
。
。
n n+n
重点来了如果1 与 2 在同一并查集,就说明1与2之间一定能找到一些线段相连,
而这些线段一定是偶数个,所以1与2一定是一伙的;
如果1与2+n相连那么同理,他俩不是一伙的;
其他情况就输出判断不了;
上代码
#include<iostream> #include<string.h> #include<stdio.h> using namespace std; const int maxa = 100005; int father[maxa*2]; int Find(int i){//printf("*"); if(father[i] == i) return i; return father[i] = Find(father[i]); } int main(){ int t, n, m; scanf("%d", &t); while(t--){ scanf("%d%d", &n, &m); for(int i = 1; i <= n*2; i++) father[i] = i; while(m--){//printf("%d ", m); char c; int a, b; //printf("*"); scanf(" %c%d%d", &c, &a, &b); //printf("%d %d", a, b); if(c == 'D'){ int fa = Find(a); int fb = Find(b+n); father[fa] = fb; fa = Find(a+n); fb = Find(b); father[fa] = fb; }else{ if(Find(a) == Find(b)) printf("In the same gang. ");//printf("*"); else if(Find(a) == Find(b+n) || Find(a+n) == Find(b)) printf("In different gangs. "); else printf("Not sure yet. "); } } } }
==================================华丽的分割线===========================================================
第二种方法:
核心代码如下
int find(int i){
if(i == father[i])
return i;
int tt = find(father[i]);
rank[i] = (rank[i] + rank[father[i]])%2;
father[i] = tt;
return father[i];
}
int uniform(int a, int b){
int fa = find(a);
int fb = find(b);
rank[fa] = (rank[a] + rank[b] +1)%2;
father[fa] = fb;
}
rank的值有两种0和1,分别代表两种帮派。
所有的根节点rank都为零
uniform部分:对a的最上级修改,如果是对a修改的话那么显然应该将a修改成rank[b]+1;而a的rank值如果为零那么a的最上级就应该就和a一样修改成rank[b]+1;
当a不等于0时就不用说了;
接下来看find部分:
边查询的时候对rank[i]修改
如果上一级rank一样,那么就修改,由于递归的特性修改当前值的时候上一级已经修改过了,所以可以保证正确性
而算法正确性的跟本保证在与每次修改都会father都会指向根结点,而且根结点的rank为零
这样就保证了如果根节点发生变化find的时候rank[I]的值只会改变一次(这句话很重要)
以上就是算法的正确性;
上代码
#include<iostream> #include<stdio.h> #include<string.h> using namespace std; const int maxa = 100005; int father[maxa]; int rank[maxa]; int find(int i){ if(i == father[i]) return i; int tt = find(father[i]); rank[i] = (rank[i] + rank[father[i]])%2; father[i] = tt; return father[i]; } int uniform(int a, int b){ int fa = find(a); int fb = find(b); rank[fa] = (rank[a] + rank[b] +1)%2; father[fa] = fb; } int main(){ int t, m, n; scanf("%d", &t); while(t--){ scanf("%d%d", &n, &m); for(int i = 1; i <=n ; i++) rank[i] = 0; for(int i = 1; i <= n; i++){ father[i] = i; } while(m--){ char c; int a, b; scanf(" %c%d%d", &c, &a, &b); if(c == 'D'){ uniform(a, b); }else{ if(n == 2){ printf("In different gangs. "); continue; } if(find(a) == find(b)) if(rank[a] == rank[b]) printf("In the same gang. "); else printf("In different gangs. "); else printf("Not sure yet. "); } } } }