码迷,mamicode.com
首页 > 其他好文 > 详细

POJ 3228 Gold Transportation 并查集

时间:2014-07-27 11:14:22      阅读:209      评论:0      收藏:0      [点我收藏+]

标签:

题目来源:POJ 3228 Gold Transportation

题意:有一些金矿要运走存储 求运送的最小最长边

思路:并查集 按边长从小到大排序 如果满足每个连通分量金矿的和小于等于存储的容量 那么就break输出

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int maxn = 210;
int f[maxn], ra[maxn], rb[maxn];

struct edge
{
	int u, v, w;
	edge(){}
	edge(int u, int v, int w): u(u), v(v), w(w){}
}e[40000];
int a[maxn];
int b[maxn];
void init(int n)
{
	for(int i = 0; i <= n; i++)
		f[i] = i, ra[i] = a[i], rb[i] = b[i];
}

int find(int x)
{
	if(x != f[x])
		return f[x] = find(f[x]);
	return f[x];
}

bool cmp(edge a, edge b)
{
	return a.w < b.w;
}

int main()
{
	int n;
	while(scanf("%d", &n) && n)
	{
		
		int sum1 = 0, sum2 = 0;
		for(int i = 1; i <= n; i++)
		{
			scanf("%d", &a[i]);
			sum1 += a[i];
		}
		for(int i = 1; i <= n; i++)
		{
			scanf("%d", &b[i]);
			sum2 += b[i];
		}
		init(n);
		int m;
		scanf("%d", &m);
		for(int i = 0; i < m; i++)
		{
			scanf("%d %d %d", &e[i].u, &e[i].v, &e[i].w);
		}
		sort(e, e+m, cmp);
		int ans = 0x7fffffff;
		for(int i = 0; i < m; i++)
		{
			int u = e[i].u;
			int v = e[i].v;
			int x = find(u);
			int y = find(v);
			if(x != y)
			{
				f[x] = y;
				ra[y] += ra[x];
				rb[y] += rb[x];
			}
			int j;
			for(j = 1; j <= n; j++)
			{
				int x = find(j);
				if(ra[x] > rb[x])
					break;
			}
			if(j > n)
			{
				ans = e[i].w;
				break;
			}
		}
		if(ans == 0x7fffffff)
		{
			puts("No Solution");
			continue;
		}
		printf("%d\n", ans);
	}
	return 0;
}


 

POJ 3228 Gold Transportation 并查集

标签:

原文地址:http://blog.csdn.net/u011686226/article/details/38167771

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!