问题 G: 战争联盟
时间限制: 1 Sec 内存限制: 128 MB提交: 26 解决: 11
[ 提交][ 状态][ 讨论版]
题目描述
我不知道第三次世界大战用的是什么武器,但我知道第四次世界大战时用的石头和木棍。
--爱因斯坦
在某个平行宇宙中,最糟糕的情况发生了,第三次世界大战一触即发!这个时候地球上的
n个国家,已经完全的分为了两个阵营,红色阵营和蓝色阵营。为了能够使得两方阵营能够和解,为了人类最终的命运不是核爆辐射,联合国正在采取各种各样的方式和手段促使双方阵营和解。
联合国首先要搞清楚一个问题,每一个国家都属于哪一方阵营。因为各方政治利益的权衡,所以绝大多数国家,并没有直接表态,自己属于哪个阵营。如今,联合国情报局获得了一些信息,每条信息包含两个国家的编号,表示这两个国家属于不同的阵营(情报正确无误)。凭借着现有这些零碎的信息,再给出任意两个国家的编号,能否确定他们属于同一方阵营呢?
假设此时地球上有
n个国家(2≤n≤10 ^ 5),国家的编号从1到n,每个国家不是属于红色阵营,就是属于蓝色正营。你将依次得到m条指令(2≤m≤10 ^ 5),这指令将只有以下两种情况:
1. D a b
a和b分别代表两个国家的编号。这一条指令的意思是a和b属于不同的阵营。
2.
A a b
a和b分别代表两个国家的编号。这一条指令的意思是,根据之前的所有指令,你要准确的判断出a和b是否属于同一个阵营。
输入
输入的第一行包含一个整数t(1 < =t < = 20),测试用例的数量。然后t组例子如下。每个测试用例以两个整数n和m开头,代表n个国家,后跟m行指令,每个行包含如上所述的一条消息。
输出
对于每一条A a b的指令,你的程序都应该根据之前得到的信息做出准确判断。
如果属于同一阵营,输出“Belong to same group.”
如果不属于同一阵营,输出“Belong to different group.”
如果不确定,输出“Not sure yet.”
样例输入
1
10 10
A 1 2
D 1 2
A 1 2
D 2 3
D 3 4
D 7 8
A 2 4
A 2 7
D 4 7
A 1 7
样例输出
Not sure yet.
Belong to different group.
Belong to same group.
Not sure yet.
Belong to same group.
并查集的推广问题,
若
D a b
D b c
可以令 a和非b(b+n)一个子集(a<---b+n)
b和非a一个子集(b<---a+n);
同理
b<---c+n;
a<---b+n<--c
则间接得到
a,b一个子集;
#include<iostream>
#include<cstdio>
using namespace std;
#define maxn 100000*2+500
int pre[maxn];
int Dp[maxn];//判断谁已经具有归属。
int N,M;
void init()
{
for(int i=0;i<N*2;i++)
{
pre[i] = i;
Dp[i] = 0;
}
}
int find(int x)
{
return pre[x] == x?x: find(pre[x]);
}
bool same(int x,int y)
{
x = find(x);
y = find(y);
return x == y;
}
void united(int x,int y)
{
x = find(x);
y = find(y);
if(x!=y&&Dp[x] < Dp[y])
{
pre[x] = y;
}else{
pre[y] = x;
if(Dp[x] == Dp[y])//如果都是0
Dp[x]++;
}
}
int main()
{
int T;
char fu;
int x,y;
cin>>T;
for(int i=0;i<T;i++)
{
scanf("%d%d",&N,&M);
getchar();
init();
for(int j=0;j<M;j++)
{
scanf("%c%d%d",&fu,&x,&y);
getchar();
if(fu=='A')
{
if(same(x,y+N)||same(x+N,y))
printf("Belong to different group.\n");
else if(same(x+N,y+N)||same(x,y))
printf("Belong to same group.\n");
else{
printf("Not sure yet.\n");
}
}else{
united(x,y+N);
united(x+N,y);
}
}
}
return 0;
}