poj 2186 Tarjan (666的牛)

【【快!题目连读5遍】
啦啦啦啦啦,下午要去保定晃一圈。

写了很长时间的一道题。哭叽叽

题目:
传送

题意:
大意就是,输入n,m
n是n头牛,m是m个描述
输入x,y;
x牛觉得y牛比较6;
输出 所有牛都觉得很牛的牛的个数【【是不是像绕口令

分析:
tarjan缩一下;
然后:在缩完的图之中找出度为0的点,>1就直接输出0【不解释】
== 1呢,就是直接找在这个点下的强连通点啦

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<stack>
using namespace std;
int n,m,tot,t,time,num,pos,cnt,y;
struct Edge{
    int from,to,next;
}edge[50001];
int low[10001],dfn[10001],instack[10001],vis[10001],strong[10001],ans[10001],deg[10001],first[10001];
stack<int> s;
void init(int u,int v)
{
    edge[tot].to=v;
    edge[tot].next=first[u];
    first[u]=tot++;
}
void tarjan(int a)
{
    low[a]=dfn[a]=++time,vis[a]=1;
    s.push(a),instack[a]=1;
    for(int i=first[a];i!=-1;i=edge[i].next){
        if(!dfn[edge[i].to]) tarjan(edge[i].to),low[a]=min(low[a],low[edge[i].to]);
        else if(instack[edge[i].to])
            low[a]=min(low[a],dfn[edge[i].to]);}
    if(low[a]== dfn[a]) {
        ++t,cnt=0;
        do{cnt++,y=s.top(),s.pop(),instack[y]=0,strong[y]=t;}while(y!=a);
        ans[t]=cnt;}
}
int main()
{
    scanf("%d%d",&n,&m); 
    int x,y;
    memset(first,-1,sizeof(first));
    while(m--)scanf("%d%d",&x,&y),init(x,y);
    for(int i=1;i<=n;i++) if(!vis[i])tarjan(i);
    for(int i=1;i<=n;i++)
        for(int j=first[i];j!=-1;j=edge[j].next)
        {
            if(strong[i]!=strong[edge[j].to])
                deg[strong[i]]++;
        }
    for(int i=1;i<=t;i++)
        if(deg[i]==0) num++,pos=i;  
    if(num > 1) printf("0");
    if(num == 1) printf("%d",ans[pos]);
}

哼唧!!!!!贴个短代码:
@Sirius_Ren

#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");
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值