题目:
描述S 城现有两座监狱,一共关押着 NNN 名罪犯,编号分别为 111 ~ NNN。他们之间的关系自然也极不和谐。
很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突。
我们用 “怨气值”(一个正整数值)来表示某两名罪犯之间的仇恨程度,怨气值越大,则这两名罪犯之间的积怨越多。
如果两名怨气值为 ccc 的罪犯被关押在同一监狱,他们俩之间会发生摩擦,并造成影响力为 ccc 的冲突事件。每年年末,警察局会将本年内监狱中的所有冲突事件按影响力从大到小排成一个列表,然后上报到 S 城 Z 市长那里。
公务繁忙的 Z 市长只会去看列表中的第一个事件的影响力,如果影响很坏,他就会考虑撤换警察局长。在详细考察了 NNN 名罪犯间的矛盾关系后,警察局长觉得压力巨大。
他准备将罪犯们在两座监狱内重新分配,以求产生的冲突事件影响力都较小,从而保住自己的乌纱帽。
假设只要处于同一监狱内的某两个罪犯间有仇恨,那么他们一定会在每年的某个时候发生摩擦。那么,应如何分配罪犯,才能使 Z 市长看到的那个冲突事件的影响力最小?这个最小值是多少?输入输入第一行为两个正整数 N(1≤N≤20000) 和 MM(1≤M≤100000),分别表示罪犯的数目以及存在仇恨的罪犯对数。接下来的 MMM 行每行为三个正整数 a,b,c,表示 a 号和 b号罪犯之间存在仇恨,其怨气值为 c。数据保证a,b≤N&0<c≤100000000且每对罪犯组合只出现一次。输出输出共一行,为 Z 市长看到的那个冲突事件的影响力。如果本年内监狱中未发生任何冲突事件,请输出 000。输入样例 14 6
1 4 2534
2 3 3512
1 2 28351
1 3 6618
2 4 1805
3 4 12884输出样例 13512来源NOIP2010
这是一道特别经典的并查集的题,有很多种解题的方式,这里我就来讲讲用虚点来做的方式。
虚点
虚点,简单地说就是相反面(如水对火)这类的东西,那我们怎么在编程里面创建虚点呢?
如上图,虚点就是:
int a[1010],n;
int x;
a[x]的虚点即为a[x+n];
思路
测试数据1:
很显然,这是一道二分图的题,但我们使用的是虚点,就不需要考虑这么多。
- 所以,首先需要排序,这样我们才能使怒气值更加小。
- 然后,如果罪犯x会和罪犯y冲突,那么就将x和y的虚点连起来,y和x的虚点连起来。
- 最后,如果x和y都连接到了同一个点的虚点,那么就可以输出他们的怒气值了,因为我们已经对怒气值排过序了,排完更大的,现在冲突的就是剩下的中最大的,直接输出就行了,如果过完了这个循环还没有找到,说明能分配完,输出0就行了。
代码:
#include <bits/stdc++.h>
using namespace std;
const int maxn = 400020;//2010*2
const int maxm = 1000010;
int n,m;struct node
{int u,v,w;
}g[maxm];bool cmp(node x,node y){return x.w>y.w;
}int fa[maxn];void init()
{for(int i = 1; i <= 2*n; i++){fa[i] = i;}
}//初始化 int get(int x)
{if(fa[x] == x) return x;int r = get(fa[x]);fa[x] = r;return r;
}//找根节点 void merge(int x, int y)
{fa[get(x)]=get(y);
}//合并 int main()
{cin >> n >> m;init();for(int i = 1; i <= m; i++){cin >> g[i].u >> g[i].v>>g[i].w;}sort(g+1,g+m+1,cmp); for (int i=1;i<=m;i++){if (get(g[i].u)!=get(g[i].v)){merge(g[i].u,g[i].v+n);merge(g[i].u+n,g[i].v);}else {cout<<g[i].w;return 0;}}cout<<0;return 0;
}