题意:
每头奶牛都梦想成为牛棚里的明星。被所有奶牛喜欢的奶牛就是一头明星奶牛。所有奶
牛都是自恋狂,每头奶牛总是喜欢自己的。奶牛之间的“喜欢”是可以传递的——如果A喜
欢B,B喜欢C,那么A也喜欢C。牛栏里共有N 头奶牛,给定一些奶牛之间的爱慕关系,请你
算出有多少头奶牛可以当明星
个人认为,tarjan算法鼻祖题
今天复习了一遍,打了一遍
dfn即dfs的顺序
low记录的是能返回到最早遍历的祖先的dfn
缩点搞搞出度就行了
若有两个以上的点出度为0
无解。。。
#include<cstdio> #include<iostream> #include<algorithm> using namespace std; #define love_nmr 0 #define nmr 50505 #define olinr 10101 int n,m; int dfn[olinr]; int low[olinr]; int ins[olinr]; int num; int du[olinr]; int cnt; int siz[olinr]; int head[olinr]; int tot; int belong[olinr]; struct node { int to,nxt; }edge[nmr]; struct sta { int ss[olinr]; int tp; void pop() { tp--; } int top() { return ss[tp]; } void push(int x) { tp++; ss[tp]=x; } bool empty() { if(!tp) return true; return false; } }s; inline int read() { char ch=getchar(); int f=1,x=0; while(!isdigit(ch)) { if(ch=='-') f=-f; ch=getchar(); } while(isdigit(ch)) { x=(x<<1)+(x<<3)+(ch^48); ch=getchar(); } return x*f; } inline void put(int x) { if(x<0) { putchar('-'); x=-x; } if(x>9) put(x/10); putchar(x%10+'0'); } inline void tarjan(int x) { s.push(x); ins[x]=true; dfn[x]=low[x]=++tot; for(int i=head[x];i;i=edge[i].nxt) { int go=edge[i].to; if(!dfn[go]) { tarjan(go); low[x]=min(low[x],low[go]); } else if(ins[go]) low[x]=min(low[x],dfn[go]); } if(dfn[x]==low[x]) { int tp=0; num++; do { tp=s.top(); s.pop(); ins[tp]=false; belong[tp]=num; siz[num]++; }while(tp!=x); } } inline void add(int from,int to) { cnt++; edge[cnt].to=to; edge[cnt].nxt=head[from]; head[from]=cnt; } int main() { n=read(); m=read(); int x,y; for(int i=1;i<=m;i++) { x=read(); y=read(); add(x,y); } for(int i=1;i<=n;i++) if(!dfn[i]) tarjan(i); for(int i=1;i<=n;i++) for(int j=head[i];j;j=edge[j].nxt) { int go=edge[j].to; if(belong[go]!=belong[i]) { du[belong[i]]++; } } int ans=0; for(int i=1;i<=num;i++) { if(!du[i]) { if(ans) { put(0); return 0; } ans=siz[i]; } } put(ans); return 0; }