XMU 1613 刘备闯三国之三顾茅庐(一) 【并查集】
1613: 刘备闯三国之三顾茅庐(一)
Time Limit: 1000 MS Memory Limit: 128 MB
Submit: 99 Solved: 29
[Submit][Status][Web Board]Description
刘备(161年-223年6月10日),字玄德,东汉末年幽州涿郡涿县,西汉中山靖王刘胜的后代。刘备一生极具传奇色彩,早年颠沛流离、备尝艰辛最终却凭借自己的谋略终成一方霸主。那么在那个风云激荡的年代,刘备又是如何从一个卖草鞋的小人物一步一步成为蜀汉的开国皇帝呢?让我们一起拨开历史的迷雾,还原一个真实的刘备。
公元207年冬至,当时驻军新野的刘备在徐庶的建议下,到南阳卧龙岗拜访诸葛亮。这是刘备第一次拜访诸葛亮的故事。三国演义记载的事,由于诸葛亮出游去了,导致刘备无功而返。然而据我考古发现,其实诸葛亮是抛出了如下的问题,想考察刘备本人的谋略、以确定其是否是自己要效忠的那个人:
由于战事频发,朝廷派兵往n(2<=n<=10000)个城镇押运粮草,但是沟通不顺畅,导致有些城镇粮草多运了,而有些城镇则少运了。由于战争的影响,一些道路被毁坏,但城镇之间仍有m(0<=m<=50000)条尚未损毁的道路,诸葛亮想知道是否有可能使得利用现有的道路,重新在城镇间运载粮草,使得每个城镇需要的粮草不多也不少?
刘备为人宽厚仁爱,但对此类问题也是束手无策,这时候就是是你表现的时候了。
Input
第一行包含两个整数n(2<=n<=10000) 和m(0<=m<=50000)。
以下n行,每一行包括一个整数,分别表示城镇0到城镇n-1多运的粮草,如果为负数,则表示少运了。保证输入中,这n个数和为0。
以下m行,每一行包含两个整数x,y(0<=x<y<=n-1),表示城镇x和城镇y存在一条道路。
Output
如果存在一种运载方案,输出POSSIBLE,否则输出IMPOSSIBLE。
Sample Input
5 3
100
-75
-25
-42
42
0 1
1 2
3 4Sample Output
POSSIBLEHINT
Source
[Submit][Status][Web Board]
题目链接:
http://acm.xmu.edu.cn/JudgeOnline/problem.php?id=1613
题目大意:
N个点M条边,每个点有一个粮草值,正为多余负为缺少,问能否通过边运输粮草使每个点粮草相等。
题目思路:
【并查集】
将被边连通的点并查集归为一类。
然后统计被归为一类的点的权值和,如果都为0就是可行,否则不可行。
1 /**************************************************** 2 3 Author : Coolxxx 4 Copyright 2017 by Coolxxx. All rights reserved. 5 BLOG : http://blog.csdn.net/u010568270 6 7 ****************************************************/ 8 #include<bits/stdc++.h> 9 #pragma comment(linker,"/STACK:1024000000,1024000000") 10 #define abs(a) ((a)>0?(a):(-(a))) 11 #define lowbit(a) (a&(-a)) 12 #define sqr(a) ((a)*(a)) 13 #define mem(a,b) memset(a,b,sizeof(a)) 14 const double EPS=1e-8; 15 const int J=10; 16 const int MOD=100000007; 17 const int MAX=0x7f7f7f7f; 18 const double PI=3.14159265358979323; 19 const int N=10004; 20 const int M=100004; 21 using namespace std; 22 typedef long long LL; 23 double anss; 24 LL aans; 25 int cas,cass; 26 int n,m,lll,ans; 27 int a[N],fa[N],sz[N],sum[N]; 28 int zhao(int aa) 29 { 30 if(fa[aa]==0 || fa[aa]==aa)return aa; 31 return fa[aa]=zhao(fa[aa]); 32 } 33 bool judge() 34 { 35 int i; 36 for(i=1;i<=n;i++) 37 { 38 if(sz[i] && sum[i]!=0)return 0; 39 } 40 return 1; 41 } 42 int main() 43 { 44 #ifndef ONLINE_JUDGE 45 freopen("1.txt","r",stdin); 46 // freopen("2.txt","w",stdout); 47 #endif 48 int i,j,k; 49 int x,y,z; 50 // for(scanf("%d",&cass);cass;cass--) 51 // for(scanf("%d",&cas),cass=1;cass<=cas;cass++) 52 // while(~scanf("%s",s)) 53 while(~scanf("%d",&n)) 54 { 55 mem(fa,0);mem(sz,0);mem(sum,0); 56 scanf("%d",&m); 57 for(i=1;i<=n;i++) 58 scanf("%d",&a[i]); 59 for(i=1;i<=m;i++) 60 { 61 scanf("%d%d",&x,&y); 62 x++;y++; 63 int fx=zhao(x); 64 int fy=zhao(y); 65 fa[fy]=fx; 66 } 67 for(i=1;i<=n;i++) 68 { 69 fa[i]=zhao(i); 70 sz[fa[i]]++; 71 sum[fa[i]]+=a[i]; 72 } 73 if(judge())puts("POSSIBLE"); 74 else puts("IMPOSSIBLE"); 75 } 76 return 0; 77 } 78 /* 79 // 80 81 // 82 */