NOIP2010T3——并查集的妙用(求无向图独立点集最大权)
关押罪犯
【问题描述】
S 城现有两座监狱,一共关押着 N 名罪犯,编号分别为 1~N。他们之间的关系自然也极
不和谐。很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突。我们用“怨
气值”(一个正整数值)来表示某两名罪犯之间的仇恨程度,怨气值越大,则这两名罪犯之
间的积怨越多。如果两名怨气值为 c 的罪犯被关押在同一监狱,他们俩之间会发生摩擦,并
造成影响力为 c 的冲突事件。
每年年末,警察局会将本年内监狱中的所有冲突事件按影响力从大到小排成一个列表,
然后上报到 S 城 Z 市长那里。公务繁忙的 Z 市长只会去看列表中的第一个事件的影响力,
如果影响很坏,他就会考虑撤换警察局长。
在详细考察了 N 名罪犯间的矛盾关系后,警察局长觉得压力巨大。他准备将罪犯们在
两座监狱内重新分配,以求产生的冲突事件影响力都较小,从而保住自己的乌纱帽。假设只
要处于同一监狱内的某两个罪犯间有仇恨,那么他们一定会在每年的某个时候发生摩擦。那
么,应如何分配罪犯,才能使 Z 市长看到的那个冲突事件的影响力最小?这个最小值是多
少?
【输入】
输入文件名为 prison.in。输入文件的每行中两个数之间用一个空格隔开。
第一行为两个正整数 N 和 M,分别表示罪犯的数目以及存在仇恨的罪犯对数。
接下来的 M 行每行为三个正整数 aj, bj, cj,表示 aj 号和 bj 号罪犯之间存在仇恨,其怨
气值为 cj。数据保证1 ≤ a j < b j ≤ N , 0 < c j ≤ 1,000,000,000 ,且每对罪犯组合只出现一
次。
【输出】
输出文件 prison.out 共 1 行,为 Z 市长看到的那个冲突事件的影响力。如果本年内监狱
中未发生任何冲突事件,请输出 0。
【输入输出样例】
prison.in | prison.out |
4 6 1 4 2534 2 3 3512 1 2 28351 1 3 6618 2 4 1805 3 4 12884 |
3512 |
【输入输出样例说明】
罪犯之间的怨气值如下面左图所示,右图所示为罪犯的分配方法,市长看到的冲突事件
影响力是 3512(由 2 号和 3 号罪犯引发)。其他任何分法都不会比这个分法更优。
对于 30%的数据有 N ≤ 15。
对于 70%的数据有 N ≤ 2000, M ≤ 50000。
对于 100%的数据有 N ≤ 20000, M ≤ 100000。
对于这道题,我们抽相出来,就是求求无向图独立点集最大权,我们贪心的去做,先把每个边排个序,然后依次分成两块,然后维护很多个并查集,使每个并查集里的元素都是在当前情况下不发生冲突的。
如果又加入一个新边,如果当前情况下都已经在一个集合里了,那么这就是正确解,如果到结束集合里的元素都不发生冲突,那么就输出0.
#include<cstdio> #include<algorithm> #define N 20000+10 #define M 100000+10 using namespace std; struct Edge{ int u,v,d; bool operator <(Edge b)const { return d>b.d; } }e[M]; int opp[N],f[N]; int n,m; int getf(int i) { if(i==f[i])return f[i]; else { return f[i]=getf(f[i]); } } int main() { scanf("%d%d",&n,&m); for(int i=1;i<=m;i++)scanf("%d%d%d",&e[i].u,&e[i].v,&e[i].d); sort(e+1,e+m+1); for(int i=1;i<=n;i++) f[i]=i; for(int i=1;i<=m;i++) { int x1=e[i].u; int x2=e[i].v; if(getf(x1)==getf(x2)) { printf("%d",e[i].d);return 0; } if(!opp[x1])opp[x1]=x2; if(!opp[x2])opp[x2]=x1; int op1=opp[x1]; int op2=opp[x2]; f[getf(op1)]=getf(x2); f[getf(op2)]=getf(x1); } printf("0"); return 0; }