POJ 2186 Popular cows(Kosaraju+强联通分量模板)

题目链接:http://poj.org/problem?id=2186

题目大意:给定N头牛和M个有序对(A,B),(A,B)表示A牛认为B牛是红人,该关系具有传递性,如果牛A认为牛B是红人,牛B认为牛C是红人,那么牛A也认为牛C是红人。求被其他所有牛认为是红牛的牛的总数。

解题思路:把所有牛看成顶点,把有序对(A,B)看成从A到B的有向边,那么题目就变成了求所有顶点都可到达的顶点的总数。我们可以得到一个结论,如果一个强连通分量里有一头牛被认为是红人,那么该强联通分量里的所有牛都是红人,这显然是正确的。由于我用的是Kosaraju求强联通分量,根据该算法性质,红牛只会在拓扑序最后的强联通分量里,我只需要找到最后一块强联通分量,取其中一个顶点,看是否所有点都可以到达即可。

 1 #include<iostream>
 2 #include<cstring>
 3 #include<algorithm>
 4 #include<vector>
 5 using namespace std;
 6 const int N=1e4+5;
 7
 8 vector<int>G[N];//图的邻接表
 9 vector<int>rG[N];//反向图的邻接表
10 vector<int>vs;//后序遍历的顺序的顶点列表
11 bool used[N];//记录点是否被访问
12 int cmp[N];//cmp[i]表示点i所属强联通分量的拓扑序
13
14 int V,E;
15
16 void addedge(int u,int v){
17     G[u].push_back(v);
18     rG[v].push_back(u);
19 }
20
21 void dfs(int v){
22     used[v]=true;
23     for(int i=0;i<G[v].size();i++){
24         if(!used[G[v][i]])
25             dfs(G[v][i]);
26     }
27     //回溯前进行标号
28     vs.push_back(v);
29 }
30
31 void rdfs(int v,int k){
32     used[v]=true;
33     //点v属于第k个强连通分量
34     cmp[v]=k;
35     for(int i=0;i<rG[v].size();i++){
36         if(!used[rG[v][i]])
37             rdfs(rG[v][i],k);
38     }
39 }
40
41 int scc(){
42     memset(used,false,sizeof(used));
43     vs.clear();
44     //第一次DFS
45     for(int i=1;i<=V;i++){
46         if(!used[i])
47             dfs(i);
48     }
49     memset(used,false,sizeof(used));
50     int k=0;//强联通分量块数
51     //第二次DFS
52     for(int i=vs.size()-1;i>=0;i--){
53         if(!used[vs[i]])
54             rdfs(vs[i],++k);
55     }
56     return k;
57 }
58
59 void solve(){
60     //获得强联通块数
61     int n=scc();
62     //统计备选解的个数
63     int u=0,num=0;
64     for(int i=1;i<=V;i++){
65         if(cmp[i]==n){
66             u=i;
67             num++;
68         }
69     }
70     //检查是否所有点可达
71     memset(used,0,sizeof(used));
72     rdfs(u,0);
73
74     for(int i=1;i<=V;i++){
75         if(!used[i]){
76             num=0;
77             break;
78         }
79     }
80     for(int i=1;i<=V;i++){
81         cout<<"i="<<i<<"  cmp="<<cmp[i]<<endl;
82     }
83     printf("%d\n",num);
84 }
85
86 int main(){
87     scanf("%d%d",&V,&E);
88     for(int i=1;i<=E;i++){
89         int u,v;
90         scanf("%d%d",&u,&v);
91         addedge(u,v);
92     }
93     solve();
94     return 0;
95 } 
时间: 2024-11-12 05:24:41

POJ 2186 Popular cows(Kosaraju+强联通分量模板)的相关文章

强连通分量分解 Kosaraju算法 (poj 2186 Popular Cows)

poj 2186 Popular Cows 题意: 有N头牛, 给出M对关系, 如(1,2)代表1欢迎2, 关系是单向的且可以传递, 即1欢迎2不代表2欢迎1, 但是如果2也欢迎3那么1也欢迎3. 求被所有牛都欢迎的牛的数量. 限制: 1 <= N <= 10000 1 <= M <= 50000 思路: Kosaraju算法, 看缩点后拓扑序的终点有多少头牛, 且要判断是不是所有强连通分量都连向它. Kosaraju算法,分拆完连通分量后,也完成了拓扑序. /*poj 2186

POJ 2186 Popular Cows --强连通分量

题意:给定一个有向图,问有多少个点由任意顶点出发都能达到. 分析:首先,在一个有向无环图中,能被所有点达到点,出度一定是0. 先求出所有的强连通分支,然后把每个强连通分支收缩成一个点,重新建图,这样,这个有向图就变成了一个有向无环图. 在这个新的图中,只需知道出度为0的点有几个即可. 如果出度为0的点超过1个,则输出0:否则输出出度为0的点所代表的那个强连通分支的分量数即可. 用Tarjan求强连通分量 代码: #include <iostream> #include <cstdio&g

POJ 2186 Popular Cows (强连通分量)

POJ 2186 Popular Cows 链接:http://poj.org/problem?id=2186 题意:每头奶牛都梦想着成为牧群中最受奶牛仰慕的奶牛.在牧群中,有N 头奶牛,1≤N≤10,000,给定M 对(1≤M≤50,000)有序对(A, B),表示A 仰慕B.由于仰慕关系具有传递性,也就是说,如果A 仰慕B,B 仰慕C,则A 也仰慕C,即使在给定的M 对关系中并没有(A, C).你的任务是计算牧群中受每头奶牛仰慕的奶牛数量. 思路:首先可以知道,在同一个强连通分量内的点一定互

POJ 2186 Popular Cows -- tarjan 缩点

链接: POJ 2186 Popular Cows 题意: 每一头牛都希望在牛群里面备受瞩目,在一个牛群中有N头牛(1<=N<=10000),你被给予M(1<=M<=50000)个关系对,形式如(A,B),这意味着A牛认为B牛比它更受欢迎,由于这种欢迎度是满足传递性的,那么若是A牛认为B牛更受欢迎,B牛认为C牛更受欢迎,那么A牛也会认为C牛更受欢迎.你的任务是计算出被所有牛受欢迎的牛的个数. 输入: 第一行两个整数 N 和 M 第2 到 M + 1 行,两个分开的数 A,B,意味着

poj 2186 Popular Cows 【强连通】

题目:poj 2186 Popular Cows 题意:n头牛,其中存在一些牛相互崇拜,具有传递性,问有多少头牛是被其他所有牛崇拜的. 分析:建立一个有向图,然后强连通缩点,之后求出度为0的点,假如存在多个,那么ans = 0,因为缩点之后如果x崇拜y,x也崇拜z,那么肯定y和z不能互相崇拜,不满足. 然后求出度为0的这个点缩点前环上有多少个点就ans AC代码: #include <cstdio> #include <vector> #include <iostream&g

Poj 2186 Popular Cows (Tarjan)

题目链接: Poj 2186 Popular Cows 题目描述: 有n只牛,牛之间存在一些关系,比如a认为b很受欢迎,b认为c很受欢迎,这样呢,a也会认为c很受欢迎,问根据给出的关系,有多少头牛被其他所有的牛都认为是受欢迎的? 解题思路: 对于一个有向无环图来说,其中有且仅有一个点出度为零,那么这个特殊的点,可以由其他任何点到达.那么接下来我们直接对所给的图进行强连通分量划分,然后把每个强连通分量看做一个点,判定出度为零的点有几个,如果有一个就输出这个点对应的强连通分量含有的节点个数,否则为零

POJ 2186 Popular Cows (强联通分量)

链接 :http://poj.org/problem?id=2186 一个联通分量里的所有的牛满足任何一个被其他牛认为是红人.强联通缩点之后 只需要找到一个且只有一个联通分量且它的出度为0 答案就是这个强联通分量点的个数. #include <algorithm> #include <iostream> #include <cstring> #include <cstdio> #include <vector> #include <stac

POJ 2186 Popular Cows (强联通)

id=2186">http://poj.org/problem? id=2186 Popular Cows Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 23819   Accepted: 9767 Description Every cow's dream is to become the most popular cow in the herd. In a herd of N (1 <= N <=

POJ 2186-Popular Cows (图论-强联通分量Korasaju算法)

题目链接:http://poj.org/problem?id=2186 题目大意:有n头牛和m对关系, 每一对关系有两个数(a, b)代表a牛认为b牛是“受欢迎”的,且这种关系具有传递性, 如果a牛认为b牛“受欢迎”, b牛认为c牛“受欢迎”, 那么a牛也认为c牛“受欢迎”. 现在想知道有多少头牛受除他本身外其他所有牛的欢迎? 解题思路:如果有两头或者多头牛受除他本身外其他所有牛的欢迎, 那么在这两头或者多头牛之中, 任意一头牛也受两头或者多头牛中别的牛的欢迎, 即这两头或者多头牛同属于一个强联