题目:这里

题意:

Description

  每一头牛的愿望就是变成一头最受欢迎的牛。现在有N头牛,给你M对整数(A,B),表示牛A认为牛B受欢迎。 这
种关系是具有传递性的,如果A认为B受欢迎,B认为C受欢迎,那么牛A也认为牛C受欢迎。你的任务是求出有多少头
牛被所有的牛认为是受欢迎的。

Input

  第一行两个数N,M。 接下来M行,每行两个数A,B,意思是A认为B是受欢迎的(给出的信息有可能重复,即有可
能出现多个A,B)

Output

  一个数,即有多少头牛被所有的牛认为是受欢迎的。

Sample Input

3 3
1 2
2 1
2 3

Sample Output

1

HINT

100%的数据N<=10000,M<=50000

我是看了强连通入门(讲的很清楚):http://www.2cto.com/kf/201606/517227.html

Kosaraju算法第一次dfs1将所有的点按拓扑排序逆序存进栈,第二次dfs2(此时是逆着方向回去搜)将整个图分成若干个强连通分量,。

对于这个题,可以观察出最后受到所有牛的欢迎的牛必定是在其中一个强连通分量里面的,所以看哪个强连通分量是其余所有变量都能达到的,也就等同于缩点后的

新图里面哪个的出度为0,如果出度为0的分量只有一个,那么该分量其中点的个数就是答案,如果出度为0的分量个数超过一个,那么没有答案,输出为0.

 #include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<vector>
#include<map>
#include<cmath>
using namespace std; const int M = 1e5 + ;
vector<int>q;
int sccno[M],sum[M],du[M],scc_cnt; struct Edge{
int to,next,from,odr;
}edge[M*];
int head1[M],head2[M],cas;
bool vis[M]; void add(int u,int v)
{
edge[++cas].next=head1[u];
edge[cas].odr=head2[v];
edge[cas].to=v;edge[cas].from=u;
head1[u]=cas;head2[v]=cas;
} void dfs1(int u)
{
for (int i=head1[u] ; i ; i=edge[i].next){
int v=edge[i].to;
if (vis[v]) continue;
vis[v]=true;
dfs1(v);
q.push_back(v);
}
} void dfs2(int u)
{
sccno[u]=scc_cnt;
sum[scc_cnt]++;
for (int i=head2[u] ; i ; i=edge[i].odr){
int v=edge[i].from;
if (sccno[v]) continue;
dfs2(v);
}
} int main()
{
int n,m;
scanf("%d%d",&n,&m);
scc_cnt=;cas=;
q.clear();
while (m--){
int x,y;
scanf("%d%d",&x,&y);
add(x,y);
}
memset(vis,false,sizeof(vis));
memset(sum,,sizeof(sum));
memset(du,,sizeof(du));
for (int i= ; i<=n ; i++)
if (vis[i]==false){
vis[i]=true;dfs1(i);
q.push_back(i);
} for (int i=n- ; i>= ; i--){
if (!sccno[q[i]]){
scc_cnt++;
// cout<<q[i]<<endl;
dfs2(q[i]);
}
} for (int i= ; i<=cas ; i++){
int x=sccno[edge[i].from],y=sccno[edge[i].to];
if (x==y) continue;
du[x]++;
}
int flag=-,ans;
for (int i= ; i<=scc_cnt ; i++)
if (!du[i]) flag++,ans=sum[i];
if (flag==) printf("%d\n",ans);
else puts("");
return ;
}

Tarjan算法链接也就讲的很清楚了

 #include<cstdio>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<stack>
using namespace std; const int M = 1e5 + ;
int head[M],cas,scc_cnt,dfs_clock;
int sccno[M],du[M],sum[M],lowlink[M],pre[M];
stack<int>s; int min(int x,int y){return x<y?x:y;} struct Edge{
int to,next,from;
}edge[M*]; void add(int u,int v)
{
edge[++cas].next=head[u];
edge[cas].to=v;edge[cas].from=u;
head[u]=cas;
} void dfs(int u)
{
pre[u]=lowlink[u]=++dfs_clock;
s.push(u);
for (int i=head[u] ; i ; i=edge[i].next){
int v=edge[i].to;
if (!pre[v]){
dfs(v);
lowlink[u]=min(lowlink[u],lowlink[v]);
}
else if (!sccno[v]){
lowlink[u]=min(lowlink[u],pre[v]);
}
}
if (lowlink[u]==pre[u]){
scc_cnt++;
for ( ; ; ){
int x=s.top();s.pop();
sccno[x]=scc_cnt;
if (x==u) break;
}
}
} int main()
{
int n,m;
scanf("%d%d",&n,&m);
cas=,dfs_clock=,scc_cnt=;
while (m--){
int x,y;
scanf("%d%d",&x,&y);
add(x,y);
}
memset(pre,,sizeof(pre));
memset(lowlink,,sizeof(lowlink));
memset(du,,sizeof(du));
memset(sum,,sizeof(sum));
for (int i= ; i<=n ; i++)
if (!pre[i]) dfs(i);
for (int i= ; i<=n ; i++)
sum[sccno[i]]++;
for (int i= ; i<=cas ; i++){
int u=sccno[edge[i].from],v=sccno[edge[i].to];
if (u==v) continue;
du[u]++;
}
int flag=,ans;
for (int i= ; i<=scc_cnt ; i++)
if (!du[i]) flag++,ans=sum[i];
if (flag==) printf("%d\n",ans);
else puts("");
return ;
}

bzoj 1051 (强连通) 受欢迎的牛的更多相关文章

  1. BZOJ 1051: [HAOI2006]受欢迎的牛( tarjan )

    tarjan缩点后, 有且仅有一个出度为0的强连通分量即answer, 否则无解 ----------------------------------------------------------- ...

  2. BZOJ 1051: [HAOI2006]受欢迎的牛(SCC)

    1051: [HAOI2006]受欢迎的牛 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 8172  Solved: 4470[Submit][Sta ...

  3. BZOJ 1051 最受欢迎的牛 解题报告

    题目直接摆在这里! 1051: [HAOI2006]受欢迎的牛 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 4438  Solved: 2353[S ...

  4. BZOJ 1051: [HAOI2006]受欢迎的牛 缩点

    1051: [HAOI2006]受欢迎的牛 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/ ...

  5. bzoj 1051: [HAOI2006]受欢迎的牛 tarjan缩点

    1051: [HAOI2006]受欢迎的牛 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2092  Solved: 1096[Submit][Sta ...

  6. BZOJ 1051: [HAOI2006]受欢迎的牛 强连通缩点

    题目链接: http://www.lydsy.com/JudgeOnline/problem.php?id=1051 题解: 强连通缩点得到DAG图,将图转置一下,对入度为零的点跑dfs看看能不能访问 ...

  7. 洛谷 P2341 BZOJ 1051 [HAOI2006]受欢迎的牛

    题目描述 每头奶牛都梦想成为牛棚里的明星.被所有奶牛喜欢的奶牛就是一头明星奶牛.所有奶 牛都是自恋狂,每头奶牛总是喜欢自己的.奶牛之间的“喜欢”是可以传递的——如果A喜 欢B,B喜欢C,那么A也喜欢C ...

  8. BZOJ 1051: [HAOI2006]受欢迎的牛

    Description 一个有向图,求所以能被别的点到达的点的个数. Sol Tarjan + 强连通分量 + 缩点. 缩点以后找强连通分量,缩点,然后当图有且仅有1个出度为1的点时,有答案. Cod ...

  9. bzoj 1051 [HAOI2006]受欢迎的牛(tarjan缩点)

    题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1051 题解:缩点之后判断出度为0的有几个,只有一个那么输出那个强连通块的点数,否者 ...

随机推荐

  1. flex-布局,轻松制作移动端网页

    Flex 布局教程 网页布局(layout)是CSS的一个重点应用. 布局的传统解决方案,基于盒状模型,依赖 display属性 + position属性 + float属性.它对于那些特殊布局非常不 ...

  2. linux pam 控制模式

    工作类别(type).流程栈(stack)和控制模式(control) Linux-PAM 工作的"类别"(type) PAM 的具体工作主要有以下四种类别(type):accou ...

  3. Linux Runlevel 启动 脚本

    Linux 操作系统自从开始启动至启动完毕需要经历几个不同的阶段,这几个阶段就叫做 Runlevel,同样,当Linux操作系统关闭时也要经历另外几个不同的 Runlevel,下面详细介绍一下 Run ...

  4. Linux:安装rstatd,报错

    [安装] 下载地址:http://heanet.dl.sourceforge.net/sourceforge/rstatd/安装:一次执行--tar -xzvf rpc.rstatd-4.0.1.ta ...

  5. Processing简明教程与Java平台移植方法

    1 Processing    1.1 Processing简介          Processing是一种具有革命前瞻性的新兴计算机语言,它的概念是在电子艺术的环境下介绍程序语言,并将电子艺术的概 ...

  6. UIImage类扩展返回一个带边框的圆形图片

    /** * 将image转换为圆型带边框的图片(最好写一个UIImage的类扩展) * * @param name 图片的名字 * @param borderWidth 外层边框的宽度 * @para ...

  7. sea.js 入门

    上个月学了 require.js 现在顺便来学学 sea.js. 对比下这两种的区别,看自己喜欢哪个,就在接下来的项目中去使用它吧. seajs中的所有 JavaScript 模块都遵循 CMD 模块 ...

  8. Mac下Call to undefined function imagettftext() 解决方案

    文章转载至Mac下Call to undefined function imagettftext()终极解决方案 安装了一套onethink程序准备调试,结果在登录页面发现验证码无法显示,单独访问验证 ...

  9. 多媒体(2):WAVE文件格式分析

    目录 多媒体(1):MCI接口编程 多媒体(2):WAVE文件格式分析 多媒体(3):基于WindowsAPI的视频捕捉卡操作 多媒体(4):JPEG图像压缩编码 多媒体(2):WAVE文件格式分析

  10. Hibernate—— 一对多 和 多对多关联关系映射(xml和注解)总结(转载)

    One to Many 映射关系 多对一单向外键关联(XML/Annotation) 一对多单向外键关联(XML/Annotation) 懒加载和积极加载 一对多双向外键关联(XML/Annotati ...