poj 2186 Popular Cows 强连通缩点

【题意】;给出一群牛中,一头牛认为另一头牛受欢迎的m对关系,关系式可以传递的,即若a认为b受欢迎,b认为c受欢迎,那么a也认为c受欢迎。求最多有多少头牛被所有的牛认为受欢迎。

【思路】:把这些关系形成的强连通图缩成点,就形成了一有向无环图,这个图里的出度为0的点(有且只有一个)就是被所有牛认为受欢迎的点,说明若这个点原来是强连通图就要求出这个强连通图里的总点数,

 1 #include<iostream>
 2 #include<stdio.h>
 3 #include<string.h>
 4 #include<stack>
 5 using namespace std;
 6 
 7 int pre[10002],low[10002],lt_num,c,scc_num[10002],scc,out0[10002],n,adj[10002],num,flag,scc_count[10002];
 8 struct E
 9 {
10     int to;
11     int next;
12 } edge[500000];
13 
14 stack <int >s;
15 
16 void add(int a,int b)
17 {
18     edge[num].to=b;
19     edge[num].next=adj[a];
20     adj[a]=num++;
21 }
22 
23 void dfs(int u)
24 {
25     int i,v;
26     pre[u]=low[u]=c++;
27     s.push (u);
28     for(i=adj[u]; i!=-1; i=edge[i].next)
29     {
30         v=edge[i].to;
31         if(!pre[v])//!scc_num[v]
32         {
33             dfs(v);
34 
35                 low[u]=min(low[u],low[v]);
36         }
37         else if(!scc_num[v])
38             low[u]=min(low[u],pre[v]);
39     }
40     if(low[u]==pre[u])   //是该连通分量的 第一个点
41     {
42         scc++;
43         while(1)
44         {
45             int t=s.top ();
46             s.pop ();
47             scc_num[t]=scc;             //scc_num[t]是第scc个强连通分量;
48             scc_count[scc]++;
49             if(t==u)
50                 break;
51         }
52     }
53 }
54 
55 int main()
56 {
57     int a,b,m,i;
58     while(~scanf("%d%d",&n,&m))
59     {
60         memset(adj,-1,sizeof(adj));
61         num=0;
62         while(m--)
63         {
64             scanf("%d%d",&a,&b);
65             add(a,b);
66         }
67         c=1;scc=0;
68         memset(scc_count,0,sizeof(scc_count));
69         memset(pre,0,sizeof(pre));
70         for(int i=1;i<=n;i++)
71             if(pre[i]==0)
72                 dfs(i);
73 
74 
75         memset(out0,0,sizeof(out0));
76         for(int u=1;u<=n;u++)
77             for(i=adj[u]; i!=-1; i=edge[i].next)
78             {
79                 int v; v=edge[i].to;
80                 if(scc_num[u]!=scc_num[v])
81                     out0[scc_num[u]]++;
82             }
83 
84         int ans=0,num=0;
85         for(int u=1;u<=scc;u++)
86             if(out0[u]==0)
87                 {
88                     num++;
89                     ans+=scc_count[u];
90                 }
91                 if(num==1)
92         printf("%d\n",ans);
93         else printf("0\n");
94     }
95     return 0;
96 }

 

转载于:https://www.cnblogs.com/assult/p/3871030.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值