Popular Cows
Time Limit: 2000MS   Memory Limit: 65536K
Total Submissions: 27496   Accepted: 11059

Description

Every cow's dream is to become the most popular cow in the herd. In a herd of N (1 <= N <= 10,000) cows, you are given up to M (1 <= M <= 50,000) ordered pairs of the form (A, B) that tell you that cow A thinks that cow B is popular. Since popularity is transitive, if A thinks B is popular and B thinks C is popular, then A will also think that C is
popular, even if this is not explicitly specified by an ordered pair
in the input. Your task is to compute the number of cows that are
considered popular by every other cow.

Input

* Line 1: Two space-separated integers, N and M

* Lines 2..1+M: Two space-separated numbers A and B, meaning that A thinks B is popular.

Output

* Line 1: A single integer that is the number of cows who are considered popular by every other cow.

Sample Input

3 3
1 2
2 1
2 3

Sample Output

1

Hint

Cow 3 is the only cow of high popularity.
 
题目:样例解释,3头牛,3条边。 1崇拜2, 2崇拜1, 2崇拜3。  3号牛能获得出自己之外的所有牛的崇拜,这样的牛有多少?输出数量。
注意:崇拜的关系是可以往上层传递的。1->2, 2->3, 可以得到:1->3.
 
通过Tarjan算法,我们会把所有的节点划分到不同的枪连通分量中。根据强连通分量的定义,处在同一个强连通分量中的节点,一会是互相仰慕的.
把每个强连通分量看成是一个节点。这样就构造成一棵树(树边是有向的),看这个树有几个出度为0的强连通分量。如果仅有一个,则会存在正确
答案。否则不可能存在满足题目的可能,只能输出0。
 
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <algorithm>
#define N 10000+10
#define M 51000+10 using namespace std;
int n, m;
struct node
{
int to;
int next;
}edge[M];//保证存下所有的边 int head[N], dfn[N], low[N], vis[N];
int stack[N], num[N], du[N];
int counter, top, cut, cnt;
/*
dfn[]:记录搜索时各顶点的访问次序,即深度优先数
low[u]:每个节点定义一个low值 low[u]是从u或u的子孙出发通过回边可以到达
的最低深度优先数
注意:dfn[]在搜索前进时进行统计 low[]值是在回退的时候进行计算
du[]:表示出度
*/
void init()
{
memset(head, -1, sizeof(head));
memset(vis, 0, sizeof(vis));
memset(dfn, 0, sizeof(dfn));
memset(low, 0, sizeof(low));
memset(num, 0, sizeof(num));
memset(du, 0, sizeof(du));
counter=1;//begin 1
cnt=0;//边 计数
top=0;// stack top
cut=0;
} void addedge(int u, int v)
{
edge[cnt].to=v;
edge[cnt].next=head[u];
head[u]=cnt++;
}//静态数组 倒着建边 void dfs(int u)
{
dfn[u]=low[u]=counter++;//counter的值从1开始
vis[u]=1;//标记访问
stack[top++]=u;
//遍历从u节点发出的边
for(int i=head[u]; i!=-1; i=edge[i].next){
int v=edge[i].to;
if(!vis[v]){//这是一条树边
dfs(v);
low[u]=min(low[u], low[v]);//回溯的时候比较
}
else if(vis[v]){
low[u]=min(low[u], dfn[v]);
}
}
if(low[u]==dfn[u]){
cut++;//表示已经找到了一个强连通分量
while(top>0 && stack[top]!=u){
top--;
vis[stack[top]]=2;//
num[stack[top]]=cut;//确定当前这个节点属于哪个强连通分量
}
}
} int main()
{
int i, j;
while(scanf("%d %d", &n, &m)!=EOF)
{
init();//进行初始化
int u, v;
for(i=1; i<=m; i++){
scanf("%d %d", &u, &v);
addedge(u, v);
}//建立单向边 for(i=1; i<=n; i++){
if(!vis[i]){
dfs(i);
}
}//不知道为什么在此处计数一下连通分量的数目 如果大于1 直接输出0 这样提交 会WA。不写就对 for(i=1; i<=n; i++){
for(j=head[i]; j!=-1; j=edge[j].next){
int v=edge[j].to;
if(num[i]!=num[v])//如果连通分量的编号不等
du[num[i]]++;//出度++
}
}
int pos; int dd=0;
for(i=1; i<=cut; i++){
if(du[i]==0){
dd++; pos=i;
}
}
//printf("pos = %d\n", pos);
if(dd==1){
int ans=0;
for(i=1; i<=n; i++){
if(num[i]==pos)
ans++;
}
printf("%d\n", ans );
}
else printf("0\n");
}
return 0;
}

使用STL-stack模拟的,注意在栈弹出节点的地方的写法,在循环跳出之后再弹出一次,因为遇到u==v的时候就跳出了,而u没有被弹出。

上面的数组模拟的stack操作也可以再修改一下。

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <stack>
#include <algorithm>
#define N 10000+10
#define M 51000+10 using namespace std;
int n, m;
struct node
{
int to;
int next;
}edge[M];//保证存下所有的边 int head[N], dfn[N], low[N], vis[N];
int num[N], du[N];
int counter, top, cut, cnt;
/*
dfn[]:记录搜索时各顶点的访问次序,即深度优先数
low[u]:每个节点定义一个low值 low[u]是从u或u的子孙出发通过回边可以到达
的最低深度优先数
注意:dfn[]在搜索前进时进行统计 low[]值是在回退的时候进行计算
du[]:表示出度
*/
stack<int>q;
void init()
{
memset(head, -1, sizeof(head));
memset(vis, 0, sizeof(vis));
memset(dfn, 0, sizeof(dfn));
memset(low, 0, sizeof(low));
memset(num, 0, sizeof(num));
memset(du, 0, sizeof(du));
counter=1;//begin 1
cnt=0;//边 计数
top=0;// stack top
cut=0;
} void addedge(int u, int v)
{
edge[cnt].to=v;
edge[cnt].next=head[u];
head[u]=cnt++;
}//静态数组 倒着建边 void dfs(int u)
{
dfn[u]=low[u]=counter++;//counter的值从1开始
vis[u]=1;//标记访问
q.push(u);
//遍历从u节点发出的边
for(int i=head[u]; i!=-1; i=edge[i].next){
int v=edge[i].to;
if(!vis[v]){//这是一条树边
dfs(v);
low[u]=min(low[u], low[v]);//回溯的时候比较
}
else if(vis[v]){
low[u]=min(low[u], dfn[v]);
}
}
if(low[u]==dfn[u]){
cut++;//表示已经找到了一个强连通分量
while(!q.empty() && q.top()!=u){
vis[q.top()]=2;//
num[q.top()]=cut;//确定当前这个节点属于哪个强连通分量
q.pop();
}
vis[q.top()]=2;//
num[q.top()]=cut;//确定当前这个节点属于哪个强连通分量
q.pop(); //还要将u节点弹出
}
} int main()
{
int i, j;
while(scanf("%d %d", &n, &m)!=EOF)
{
init();//进行初始化
int u, v;
for(i=1; i<=m; i++){
scanf("%d %d", &u, &v);
addedge(u, v);
}//建立单向边 int cur=0;
for(i=1; i<=n; i++){
if(!vis[i]){
dfs(i); cur++;
}
} for(i=1; i<=n; i++){
for(j=head[i]; j!=-1; j=edge[j].next){
int v=edge[j].to;
if(num[i]!=num[v])//如果连通分量的编号不等
du[num[i]]++;//出度++
}
}
int pos; int dd=0;
for(i=1; i<=cut; i++){
if(du[i]==0){
dd++; pos=i;
}
}
if(dd==1){
int ans=0;
for(i=1; i<=n; i++){
if(num[i]==pos)
ans++;
}
printf("%d\n", ans );
}
else printf("0\n");
}
return 0;
}

poj 2186 Popular Cows 【强连通分量Tarjan算法 + 树问题】的更多相关文章

  1. poj 2186 Popular Cows (强连通分量+缩点)

    http://poj.org/problem?id=2186 Popular Cows Time Limit: 2000MS   Memory Limit: 65536K Total Submissi ...

  2. POJ 2186 Popular Cows(强连通分量缩点)

    题目链接:http://poj.org/problem?id=2186 题目意思大概是:给定N(N<=10000)个点和M(M<=50000)条有向边,求有多少个“受欢迎的点”.所谓的“受 ...

  3. POJ 2186 Popular Cows --强连通分量

    题意:给定一个有向图,问有多少个点由任意顶点出发都能达到. 分析:首先,在一个有向无环图中,能被所有点达到点,出度一定是0. 先求出所有的强连通分支,然后把每个强连通分支收缩成一个点,重新建图,这样, ...

  4. POJ 2186 Popular Cows 强连通分量模板

    题意 强连通分量,找独立的块 强连通分量裸题 #include <cstdio> #include <cstdlib> #include <cstring> #in ...

  5. POJ2186 Popular Cows 强连通分量tarjan

    做这题主要是为了学习一下tarjan的强连通分量,因为包括桥,双连通分量,强连通分量很多的求法其实都可以源于tarjan的这种方法,通过一个low,pre数组求出来. 题意:给你许多的A->B ...

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

    poj 2186 Popular Cows 题意: 有N头牛, 给出M对关系, 如(1,2)代表1欢迎2, 关系是单向的且能够传递, 即1欢迎2不代表2欢迎1, 可是假设2也欢迎3那么1也欢迎3. 求 ...

  7. tarjan缩点练习 洛谷P3387 【模板】缩点+poj 2186 Popular Cows

    缩点练习 洛谷 P3387 [模板]缩点 缩点 解题思路: 都说是模板了...先缩点把有环图转换成DAG 然后拓扑排序即可 #include <bits/stdc++.h> using n ...

  8. POJ 2186 Popular Cows (强联通)

    id=2186">http://poj.org/problem? id=2186 Popular Cows Time Limit: 2000MS   Memory Limit: 655 ...

  9. 有向图强连通分量Tarjan算法

    在https://www.byvoid.com/zhs/blog/scc-tarjan中关于Tarjan算法的描述非常好,转述如下: 首先解释几个概念: 有向图强连通分量:在有向图G中,如果两个顶点间 ...

随机推荐

  1. 【转载】C#时间差的计算,精确输出“年月天时分秒”

    ======================== 感谢“不忘初心”大神的分享======================== 原博地址:http://www.cnblogs.com/IT-Bear/a ...

  2. COGS 693. [SDOI2005]Antiprime数 唯一分解定理逆用

    693. Antiprime数 ★★   输入文件:antip.in   输出文件:antip.out   简单对比 时间限制:1 s   内存限制:128 MB 如果一个自然数n(n>=1), ...

  3. Spring JDBC-混合框架的事务管理

    ​ Spring 抽象的 DAO 体系兼容多种数据访问技术,它们各有特色,各有千秋. Hibernate 是非常优秀的 ORM 实现方案,但对底层 SQL 的控制不太方便 MyBatis 则通过模板化 ...

  4. linux磁盘清理

    一.背景: 1.由于linux系统空间是由挂载磁盘得来的,但有时装系统时挂载/根目录空间不大,现仅清除用户下载的大文件 二.方法: 1.输入命令df -h显示当前磁盘挂载(包含剩余空间)情况这里写图片 ...

  5. js 自学,云知梦知识 点理论

    一.第1章(1--4) 何为js特效 1.定义事件(触发时机+行为) 2.触发事件(行为发生) 3.事件发生是具有重复性   js是基本对象的语言. 面向对像编程 1.如何获得该对象 2.如何 调用该 ...

  6. js实现还可输入多少个字

    // 还可输入的字数个数 function checkLength(obj, len) { var maxChars = len; if (obj.value.length <= maxChar ...

  7. cxGrid时间格式与导出Excel

    引用cxFormats单元: ShortDateFormat := 'dd/mm/yyyy'; DateSeparator := '/'; cxFormatController.UseDelphiDa ...

  8. HTML 之 Table 表格详解

    HTML 之 Table 表格详解 HTML中的table可以大致分为三个部分: thead ---------表格的页眉 tbody ---------表格的主体 tfoot ---------定义 ...

  9. Python3.6全栈开发实例[005]

    5.接收两个数字参数,返回比较大的那个数字. def compare(a,b): return a if a > b else b # 三元表达式 print(compare(20,100))

  10. Servlet 运行原理

    一:servlet定义 Servlet是一个Java应用程序,运行在服务器端,用来处理客户端请求并作出响应的程序. 二:简单servlet实例 //导入所需的包 import javax.servle ...