罪犯"/>
【二分】关押罪犯
题目:关押罪犯(prison.pas/cpp/in/out)
题目描述
S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N。他们之间的关系自然也极不和谐。很多罪犯之间甚至积怨已久,如果客观条件具备则随时可能爆发冲突。我们用“怨气值”(一个正整数值)来表示某两名罪犯之间的仇恨程度,怨气值越大,则这两名罪犯之间的积怨越多。如果两名怨气值为c 的罪犯被关押在同一监狱,他们俩之间会发生摩擦,并造成影响力为c 的冲突事件。
每年年末,警察局会将本年内监狱中的所有冲突事件按影响力从大到小排成一个列表,然后上报到S 城Z 市长那里。公务繁忙的Z 市长只会去看列表中的第一个事件的影响力,如果影响很坏,他就会考虑撤换警察局长。
在详细考察了N 名罪犯间的矛盾关系后,警察局长觉得压力巨大。他准备将罪犯们在两座监狱内重新分配,以求产生的冲突事件影响力都较小,从而保住自己的乌纱帽。假设只要处于同一监狱内的某两个罪犯间有仇恨,那么他们一定会在每年的某个时候发生摩擦。那么,应如何分配罪犯,才能使Z 市长看到的那个冲突事件的影响力最小?这个最小值是多少?
输入格式
输入文件的每行中两个数之间用一个空格隔开。
第一行为两个正整数N 和M,分别表示罪犯的数目以及存在仇恨的罪犯对数。
接下来的M 行每行为三个正整数aj,bj,cj,表示aj 号和bj 号罪犯之间存在仇恨,其怨
气值为cj。数据保证1<aj=<=bj<=N ,0 < cj≤ 1,000,000,000,且每对罪犯组合只出现一
次。
输出格式
共1 行,为Z 市长看到的那个冲突事件的影响力。如果本年内监狱中未发生任何冲突事件,请输出0。
样例输入
4 6
1 4 2534
2 3 3512
1 2 28351
1 3 6618
2 4 1805
3 4 12884
样例输出
3512
【输入输出样例说明】
罪犯之间的怨气值如下面左图所示,右图所示为罪犯的分配方法,市长看到的冲突事件影响力是3512(由2 号和3 号罪犯引发)。其他任何分法都不会比这个分法更优。
一开始错了主要是因为忘记了,删除一些边之后,可能会出现多个连通分量,必须保证每一个连通分量都满足要求。
#include <string>
#include <cstdio>
#include <algorithm>
#define MAX(a,b) ((a)>(b)?(a):(b))
#define MIN(a,b) ((a)<(b)?(a):(b))using std::sort;
long n,m,mid;struct Edge
{long f;long t;long c;bool operator<(const Edge& e2)const {return f<e2.f;}
};bool used[20010];
bool sign[20010];
Edge edge[200010];
long start[20010];
long cque[200010];
long que[2000010];
long cedge=0;
const long inf = 0x7f7f7f7f;bool can()
{memset(sign,0,sizeof sign);memset(used,0,sizeof used);for (long i=1;i<n+1;i++){if (used[i]) continue;long l=0,r=0;que[++r] = i;sign[r] = true;used[r] = true;while (l < r){long u = que[++l];for (long vv=start[u];edge[vv].f==u;vv++){long v = edge[vv].t;if (edge[vv].c > mid){if (used[v]){if (sign[v]==sign[u]){return false;}}else{used[v] = true;sign[v] = !sign[u];que[++r] = v;}}}}}return true;
}
/*
bool can(long u)
{for (long vv=start[u];edge[vv].f==u;vv++){long v = edge[vv].t;if (edge[vv].c>mid){if (used[v]){if (sign[v]==sign[u])return false;}else{used[v] = true;sign[v] = !sign[u];if (!can(v))return false;}}}return true;
}*/long getint()
{char tmp;long rs=0;bool sgn=1;do tmp=getchar();while (!isdigit(tmp)&&tmp-'-');if (tmp=='-'){tmp=getchar();sgn=0;}do rs=(rs<<3)+(rs<<1)+tmp-'0';while (isdigit(tmp=getchar()));return sgn?rs:-rs;
}int main()
{freopen("prison.in","r",stdin);freopen("prison.out","w",stdout);n = getint();m = getint();for (long i=1;i<m+1;i++){long a = getint();long b = getint();long c = getint();++cedge;edge[cedge].f = a;edge[cedge].t = b;edge[cedge].c = c;++cedge;edge[cedge].f = b;edge[cedge].t = a;edge[cedge].c = c;cque[i] = c;}sort(edge+1,edge+cedge+1);edge[cedge+1].f = inf;sort(cque+1,cque+1+m);for (long i=1;i<cedge+1;i++)if (!start[edge[i].f])start[edge[i].f] = i;long l=0,r=m;long ans = inf;while (l <= r){memset(used,0,sizeof used);memset(sign,0,sizeof sign);used[1] = true;sign[1] = true;long mmid = (l+r)>>1;mid = cque[mmid];if (can()){if (ans > mid)ans = mid;r = mmid-1; }else{l = mmid+1; }}printf("%ld\n",ans);return 0;
}
更多推荐
【二分】关押罪犯
发布评论