http://poj.org/problem?id=2186
给你n头牛 他们之间会有一种 A认为B popular 的关系 而且这种关系是可传递的
问你有多少头牛是被所有其他的牛认为popular
1,缩点 将一个联通分量内的点缩成一个
2,缩点同时记录联通分量内点的个数
3,缩点后图变成了树 或者是 (非树)森林 如果是树 则树根连通分量内的点就是答案 如果是森林 答案为0
生成的树或森林 根方向上是被指向的。
4,由于3的原因 搜索时还有注意标记哪些缩点是没有可能成为根的缩点
5,如果根缩点只有一个(树) 则根就是答案 如果多个(非树森林)则为0
注意 Tarjan 搜索是并不一定一次就能将所有的点搜干净 注意这一点 想办法处理
代码及其注释:
#include<iostream> #include<cstring> #include<stack> #include<cstdio> #include<queue> using namespace std; const int N=10005; struct node { struct tt *next; }mem[N]; struct tt { struct tt *next; int j; }; int deep; int low[N]; int dfn[N]; bool visited[N]; bool in[N]; stack< int >str; int uppoint[N]; int th[N]; //第几次用Tarjan搜索 bool f[N]; //如果此点为其它缩点的父结点 所在联通分量不可能是根 int sum[N]; void build( int i, int j) { struct tt *t= new tt; t->j=j; t->next=mem[i].next; mem[i].next=t; } void Clearlist( int n) { struct tt *t; for ( int i=1;i<=n;++i) { while (mem[i].next!=NULL) { t=mem[i].next; mem[i].next=t->next; delete t; } } } void Tarjan( int pre, int x, int ith) { ++deep; dfn[x]=low[x]=deep; th[x]=ith; //第几次使用Tarjan visited[x]= true ; in[x]= true ; str.push(x); struct tt *t=mem[x].next; bool think= false ; while (t!=NULL) { if (visited[t->j]== false ) { Tarjan(x,t->j,ith); low[x]=min(low[x],low[t->j]); } else if (in[t->j]== true ) { low[x]=min(low[x],dfn[t->j]); } else if (th[t->j]<ith) //如果出现前几次用Tarjan搜过的点 则x所在的联通分量也不可能是根 { think= true ; } t=t->next; } if (think) { f[x]= true ; } if (low[x]==dfn[x]) { int num=0; while (str.top()!=x) { uppoint[str.top()]=x; //缩点 in[str.top()]= false ; ++num; str.pop(); } uppoint[str.top()]=x; in[str.top()]= false ; ++num; str.pop(); sum[x]=num; f[pre]= true ; //前一个点为父结点 不可能是根 } } int main() { int n,m; while ( scanf ( "%d %d" ,&n,&m)!=EOF) { while (m--) { int i,j; scanf ( "%d %d" ,&i,&j); build(i,j); } memset (th,-1, sizeof (th)); memset (dfn,-1, sizeof (dfn)); memset (visited, false , sizeof (visited)); memset (in, false , sizeof (in)); memset (f, false , sizeof (in)); memset (uppoint,-1, sizeof (uppoint)); memset (sum,0, sizeof (sum)); deep=0; for ( int i=1,t=1;i<=n;++i) { if (dfn[i]==-1) //防止一次搜不完 t 是第几次搜 Tarjan(0,i,t++); } for ( int i=1;i<=n;++i) { if (f[i]) //把不可能是根联通分量的点的标记转移的缩点上 f[uppoint[i]]= true ; } int num=0; int k=0; for ( int i=0;i<=n;++i) { if (i==uppoint[i]&&!f[i]) { ++num; k=i; } } if (num!=1) //不只一个根联通分量 { printf ( "0\n" ); } else { printf ( "%d\n" ,sum[k]); } Clearlist(n); } return 0; } |
分类:
连通分量
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】博客园携手 AI 驱动开发工具商 Chat2DB 推出联合终身会员
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 为什么 .NET8线程池 容易引发线程饥饿
· golang自带的死锁检测并非银弹
· 如何做好软件架构师
· 记录一次线上服务OOM排查
· Linux实时系统Xenomai宕机问题的深度定位过程
· 2025年广告第一单,试试这款永久免费的开源BI工具
· 为什么 .NET8线程池 容易引发线程饥饿
· 场景题:假设有40亿QQ号,但只有1G内存,如何实现去重?
· 在 .NET 中使用 Tesseract 识别图片文字
· .NET 响应式编程 System.Reactive 系列文章(一):基础概念