Summer Holiday
Time Limit: 10000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 1190 Accepted Submission(s): 529
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计算出至少要通知多少人,至少得花多少电话费就能让所有人都被通知到吗?
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。
第一行两个整数N和M(1<=N<=1000, 1<=M<=2000),表示人数和联系对数。
接下一行有N个整数,表示Wiskey联系第i个人的电话费用。
接着有M行,每行有两个整数X,Y,表示X能联系到Y,但是不表示Y也能联系X。
Output
输出最小联系人数和最小花费。
每个CASE输出答案一行。
每个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思路:先缩点。然后找入度为0的强连通分量,只需要打电话给该强连通分量中花费最小的一个人即可。AC代码:#include <iostream> #include <cstdio> #include <cstring> #include <string> #include <algorithm> #include <queue> #include <vector> #include <cmath> #include <cstdlib> #define L(rt) (rt<<1) #define R(rt) (rt<<1|1) using namespace std; const int maxn=1005; const int maxm=2005; const int INF=1000000000; struct node { int v,next; } edge[maxm]; int head[maxn],low[maxn],dfn[maxn],cost[maxn],mincost[maxn]; int stack[maxn],scc[maxn],in[maxn],out[maxn]; bool ins[maxn]; int n,m,top,cnt,snum,num; void init() { memset(head,-1,sizeof(head)); num=0; } void add(int u,int v) { edge[num].v=v; edge[num].next=head[u]; head[u]=num++; } void input() { int a,b; for(int i=1;i<=n;i++) scanf("%d",&cost[i]); while(m--) { scanf("%d%d",&a,&b); add(a,b); } } void dfs(int u) { int x; dfn[u]=low[u]=++cnt; stack[top++]=u; ins[u]=true; for(int i=head[u];i!=-1;i=edge[i].next) { int v=edge[i].v; if(!dfn[v]) { dfs(v); low[u]=min(low[u],low[v]); } else if(ins[v]) low[u]=min(low[u],dfn[v]); } if(low[u]==dfn[u]) { ++snum; do{ x=stack[--top]; ins[x]=false; scc[x]=snum; if(cost[x]<mincost[snum]) mincost[snum]=cost[x]; }while(x!=u); } } void tarjan() { memset(dfn,0,sizeof(dfn)); memset(ins,false,sizeof(ins)); for(int i=1;i<=n;i++) mincost[i]=INF; snum=cnt=top=0; for(int i=1;i<=n;i++) if(!dfn[i]) dfs(i); } void solve() { memset(in,0,sizeof(in)); memset(out,0,sizeof(out)); for(int u=1;u<=n;u++) for(int i=head[u];i!=-1;i=edge[i].next) { int v=edge[i].v; if(scc[u]!=scc[v]) in[scc[v]]++; } int ans1=0,ans2=0; for(int i=1;i<=snum;i++) if(!in[i]) { ans1++; ans2+=mincost[i]; } printf("%d %d\n",ans1,ans2); } int main() { while(~scanf("%d%d",&n,&m)) { init(); input(); tarjan(); solve(); } return 0; }