POJ 2186 Tarjan

题意:有n(n<=10000)头牛,每头牛都想成为最受欢迎的牛,给出m(m<=50000)个关系,如(1,2)代表1欢迎2,关系可以传递,但是不是相互的,那么就是说1欢迎2不代表2欢迎1,但是如果2欢迎3那么1也欢迎3.
输入第一行为n,m第2到1+m行为m个欢迎关系,求被所有牛都欢迎的牛的数量。
思路:Tarjan求强联通分量做。
1.如果图不联通,直接输出零。(不解释)
2.如果有超过1个出度=0的点,直接输出零。因为它肯定不是最受欢迎的牛。
3.如果只有一个出度等于零的点,那它的强联通分量里的所有点都是最受欢迎的牛。
题目在这里

#include <stack>
#include <cstdio>
#include <vector>
#include <algorithm>
using namespace std;
int low[10005],dfn[10005],n,m,cnt=0,t=0,p[10005],out[10005],ans=0;
bool vis[10005];
stack<int>stk;
vector<int>v[10005];
void tarjan(int x)
{
    low[x]=dfn[x]=++cnt,vis[x]=1,stk.push(x);
    for(int i=0;i<v[x].size();i++)
        if(!dfn[v[x][i]])
            tarjan(v[x][i]),low[x]=min(low[v[x][i]],low[x]);
        else if(vis[v[x][i]])
            low[x]=min(low[x],dfn[v[x][i]]);
    if(dfn[x]==low[x]){
        int y;t++;
        do y=stk.top(),stk.pop(),vis[y]=0,p[y]=t;while(y!=x);
    }
}
int main()
{
    register int x,y,q;
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++)
        scanf("%d%d",&x,&y),v[x].push_back(y);
    for(int i=1;i<=n;i++)if(!dfn[i])tarjan(i);
    for(int i=1;i<=n;i++)
        for(int j=0;j<v[i].size();j++)
            if(p[v[i][j]]!=p[i])out[p[i]]++;
    for(int i=1;i<=t;i++)
        if(!out[i])ans++,q=i;
    if(ans==1){
        for(int i=1;i<=n;i++)if(p[i]==q)ans++;
        printf("%d",ans-1);
    }
    else puts("0");
}

这里写图片描述

你可能感兴趣的:(poj,Tarjan)