洛谷-P1525 [NOIP2010 提高组] 关押罪犯
题目描述
S 城现有两座监狱,一共关押着 N 名罪犯,编号分别为1−N。他们之间的关系自然也极不和谐。很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突。我们用“怨气值”(一个正整数值)来表示某两名罪犯之间的仇恨程度,怨气值越大,则这两名罪犯之间的积怨越多。如果两名怨气值为 cc 的罪犯被关押在同一监狱,他们俩之间会发生摩擦,并造成影响力为 cc 的冲突事件。
每年年末,警察局会将本年内监狱中的所有冲突事件按影响力从大到小排成一个列表,然后上报到 S 城 Z 市长那里。公务繁忙的 Z 市长只会去看列表中的第一个事件的影响力,如果影响很坏,他就会考虑撤换警察局长。
在详细考察了N 名罪犯间的矛盾关系后,警察局长觉得压力巨大。他准备将罪犯们在两座监狱内重新分配,以求产生的冲突事件影响力都较小,从而保住自己的乌纱帽。假设只要处于同一监狱内的某两个罪犯间有仇恨,那么他们一定会在每年的某个时候发生摩擦。
那么,应如何分配罪犯,才能使 Z 市长看到的那个冲突事件的影响力最小?这个最小值是多少?
输入格式
每行中两个数之间用一个空格隔开。第一行为两个正整数 N,M,分别表示罪犯的数目以及存在仇恨的罪犯对数。接下来的 M 行每行为三个正整数aj,bj,cj,表示 aj 号和bj 号罪犯之间存在仇恨,其怨气值为 cj。数据保证 1<aj≤bj≤N, 0<cj≤1e9,且每对罪犯组合只出现一次。
输出格式
共 11 行,为 Z 市长看到的那个冲突事件的影响力。如果本年内监狱中未发生任何冲突事件,请输出 0。
输入输出样例
输入 #1 |
输出 #1 |
4 61 4 25342 3 35121 2 283511 3 66182 4 18053 4 12884 |
3512 |
说明/提示
【输入输出样例说明】
罪犯之间的怨气值如下面左图所示,右图所示为罪犯的分配方法,市长看到的冲突事件影响力是 35123512(由 22 号和 33 号罪犯引发)。其他任何分法都不会比这个分法更优。
【数据范围】
对于 30%的数据有 N≤15。
对于 70% 的数据有 50000N≤2000,M≤50000。
对于 100% 的数据有100000N≤20000,M≤100000。
题目分析
囚犯之间有怨气值,怨气值越高造成破坏越大。我们要将其分散到两个不同监狱,最优情况肯定是把怨气值大的优先分开。
所以首先按怨气值,对囚犯对们降序排序,然后将囚犯对分开分到两个集合,遇到一对囚犯在两个集合中都会发生矛盾的时候(他们“祖宗”相同),他们这一对囚犯的怨气值就造成最大破坏。
可行代码
Vector不定长数组,知识快捷入口。Operator运算符重载,知识快捷入口。
#include <algorithm> #include <iostream> #include <vector> using namespace std; const int MAX = 2e6 + 10; int fa[MAX]; struct Prisoner { int pri1, pri2; int value; bool operator<(Prisoner &a) const { return this->value > a.value; } // 结构体内排序(作用类似于cmp),运算符重载实现排序 }; vector<Prisoner> people; // 囚犯对及其怨气值 数组 int priNum, priPair; // 囚犯数,积怨囚犯对数 void init() { // 1->n为一个集合 n+1 -> 2n为一个集合 for (int i = 0; i < priNum * 2 + 1; i++) fa[i] = i; people.clear(); return; } int find(int aim) { if (aim != fa[aim]) return fa[aim] = find(fa[aim]); return aim; } void merge(int a, int b) { a = find(a), b = find(b); if (a != b) fa[b] = a; return; } int main() { cin >> priNum >> priPair; init(); for (int i = 0; i < priPair; i++) { Prisoner p; cin >> p.pri1 >> p.pri2 >> p.value; people.push_back(p); // 将积怨囚犯对加入数组 } sort(people.begin(), people.end()); for (int i = 0; i < people.size(); i++) { int person1 = find(people[i].pri1); // 找祖宗 int person2 = find(people[i].pri2); if (person1 != person2) { merge(people[i].pri1, people[i].pri2 + priNum); merge(people[i].pri2, people[i].pri1 + priNum); } else { cout << people[i].value << endl; // 祖宗相同,发生矛盾 return 0; } } cout << '0' << endl; return 0; }
END 感谢reader's阅读,洛谷题目链接:P1525 [NOIP2010 提高组] 关押罪犯 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)