hdu1827之強聯通
阿新 • • 發佈:2017-05-13
san esp tin cpp script auto courier 話費 -a
Total Submission(s): 1430 Accepted Submission(s): 645
Problem Description To see a World in a Grain of Sand
And a Heaven in a Wild Flower,
Hold Infinity in the palm of your hand
And Eternity in an hour.
—— William Blake
聽說lcy幫大家預定了新馬泰7日遊,Wiskey真是高興的夜不能寐啊。他想著得快點把這消息告訴大家,盡管他手上有全部人的聯系方式,可是一個一個聯系過去實在太耗時間和電話費了。他知道其它人也有一些別人的聯系方式,這樣他能夠通知其它人。再讓其它人幫忙通知一下別人。
Input 多組測試數組,以EOF結束。
第一行兩個整數N和M(1<=N<=1000, 1<=M<=2000)。表示人數和聯系對數。
接下一行有N個整數,表示Wiskey聯系第i個人的電話費用。
接著有M行,每行有兩個整數X,Y,表示X能聯系到Y,可是不表示Y也能聯系X。
Output 輸出最小聯系人數和最小花費。
每一個CASE輸出答案一行。
Sample Input
Sample Output
Summer Holiday
Time Limit: 10000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 1430 Accepted Submission(s): 645
Problem Description To see a World in a Grain of Sand
And a Heaven in a Wild Flower,
Hold Infinity in the palm of your hand
And Eternity in an hour.
—— William Blake
聽說lcy幫大家預定了新馬泰7日遊,Wiskey真是高興的夜不能寐啊。他想著得快點把這消息告訴大家,盡管他手上有全部人的聯系方式,可是一個一個聯系過去實在太耗時間和電話費了。他知道其它人也有一些別人的聯系方式,這樣他能夠通知其它人。再讓其它人幫忙通知一下別人。
你能幫Wiskey計算出至少要通知多少人。至少得花多少電話費就能讓全部人都被通知到嗎?
Input 多組測試數組,以EOF結束。
第一行兩個整數N和M(1<=N<=1000, 1<=M<=2000)。表示人數和聯系對數。
接下一行有N個整數,表示Wiskey聯系第i個人的電話費用。
接著有M行,每行有兩個整數X,Y,表示X能聯系到Y,可是不表示Y也能聯系X。
Output 輸出最小聯系人數和最小花費。
每一個CASE輸出答案一行。
Sample Input
12 16 2 2 2 2 2 2 2 2 2 2 2 2 1 3 3 2 2 1 3 4 2 4 3 5 5 4 4 6 6 4 7 4 7 12 7 8 8 7 8 9 10 9 11 10
Sample Output
3 6
#include <iostream> #include <cstdio> #include <cstdlib> #include <cstring> #include <string> #include <queue> #include <algorithm> #include <map> #include <cmath> #include <iomanip> #define INF 99999999 typedef long long LL; using namespace std; const int MAX=1000+10; int n,m,size,top,index; int head[MAX],val[MAX],dfn[MAX],low[MAX]; int mark[MAX],stack[MAX]; struct Edge{ int v,next; Edge(){} Edge(int V,int NEXT):v(V),next(NEXT){} }edge[MAX*2]; void Init(int num){ for(int i=0;i<=num;++i)head[i]=-1,mark[i]=0; size=top=index=0; } void InsertEdge(int u,int v){ edge[size]=Edge(v,head[u]); head[u]=size++; } void tarjan(int u){ if(mark[u])return; dfn[u]=low[u]=++index; stack[++top]=u; mark[u]=1; for(int i=head[u];i != -1;i=edge[i].next){ int v=edge[i].v; tarjan(v); if(mark[v] == 1)low[u]=min(low[u],low[v]); } if(dfn[u] == low[u]){ while(stack[top] != u){ mark[stack[top]]=-1; val[u]=min(val[u],val[stack[top]]); low[stack[top--]]=low[u]; } mark[u]=-1; --top; } } int main(){ int u,v; while(~scanf("%d%d",&n,&m)){ Init(n); for(int i=1;i<=n;++i)scanf("%d",val+i); for(int i=0;i<m;++i){ scanf("%d%d",&u,&v); InsertEdge(u,v); } for(int i=1;i<=n;++i){ if(mark[i])continue; tarjan(i); } for(int i=0;i<=n;++i)mark[i]=0; for(int i=1;i<=n;++i){ for(int j=head[i];j != -1;j=edge[j].next){ v=edge[j].v; if(low[i] == low[v])continue; mark[low[v]]=1; } } int sum=0,ans=0; for(int i=1;i<=n;++i){ if(!mark[low[i]] && dfn[i] == low[i])++ans,sum+=val[i]; mark[low[i]]=1; } printf("%d %d\n",ans,sum); } return 0; }
hdu1827之強聯通