POJ 2186 Popular Cows(初学强连通)

题意:有n个牛,m个有向“仰慕”关系,关系可传递,求多少个牛被所有牛都“仰慕”

思路:显然被所有牛仰慕的牛群是一强连通分量

所以先把乱图缩点成有向无环图

对有向无环图有这个重要结论:

任何连通的图都至少有一个入度为0的点和至少有一个出度为0的点(也就是至少有一个最高强连通分量和一个最低强连通分量)

所以本题是找连通图的唯一的最低强连通分量,显然仅有一个出度为0的强连通分量即是解

Tarjan算法:

//832K	79MS
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;

const int N= 1e4+5;
const int M= 5e4+50;
struct edge
{
    int u,v;
    int next;
}es[M];
int head[N];
int low[N],dfn[N],tmp[N];
int n,m;
int col[N];
int sta[N];
int top;
int scc_num;
void tardfs(int u,int lay)
{
    tmp[u]=1;
    low[u]=lay;
    dfn[u]=lay;
    sta[++top]=u;
    for(int i=head[u];~i;i=es[i].next)
    {
        int v=es[i].v;
        if(tmp[v]==0) tardfs(v,lay+1);
        if(tmp[v]==1) low[u]=min(low[u],low[v]);
    }
    if(dfn[u]==low[u])
    {
        ++scc_num;
        while(1)
        {
            int x=sta[top];
            low[x]=scc_num;
            tmp[x]=2;
            if(sta[top--]==u) break;
        }
    }
}
int tarjan()
{
    for(int i=1;i<=n;i++)
        if(tmp[i]==0) tardfs(i,1);
    return scc_num;
}
void ini()
{
    memset(head,-1,sizeof(head));
    memset(tmp,0,sizeof(tmp));
    memset(col,0,sizeof(col));
    memset(low,0,sizeof(low));
    top=0;
    scc_num=0;
}
int main()
{
    while(~scanf("%d%d",&n,&m))
    {
        ini();
        for(int i=1;i<=m;i++)
        {
            int  u,v;
            scanf("%d%d",&u,&v);
            es[i].u=u;
            es[i].v=v;
            es[i].next=head[u];
            head[u]=i;
        }
        int num=tarjan();
        int cnt=0;
        for(int i=1;i<=m;i++)
        {
            if(low[es[i].u]==low[es[i].v]) continue;
            int u=es[i].u;
            col[low[u]] = 1;
        }
        int t,ans=0;
        for(int i=1;i<=num;i++) if(col[i]==0) cnt++,t=i;
        if(cnt!=1) puts("0");
        else
        {
            for(int i=1;i<=n;i++)
                if(low[i]==t) ans++;
            printf("%d\n",ans);
        }
    }
    return 0;
}

Garbow算法:

//856K	32MS	C++	1844B
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std;

const int N= 1e4+5;
const int M= 5e4+50;
struct edge
{
    int u,v;
    int next;
}es[M];
int head[N];
int dfn[N];
int n,m;
int gro[N];
int col[N];
int sta1[N];
int sta2[N];
int tp1,tp2;
int scc_num;

void garbow(int u,int lay)
{
    dfn[u]=lay;
    sta1[++tp1]=u;
    sta2[++tp2]=u;
    for(int i=head[u];~i;i=es[i].next)
    {
        int v=es[i].v;
        if(gro[v]) continue;
        if(!dfn[v]) garbow(v,lay+1);
        else while(dfn[sta1[tp1]]>dfn[v]) tp1--;
    }

    if(sta1[tp1]==u)
    {
        tp1--;
        scc_num++;
        while(1)
        {
            int v=sta2[tp2];
            gro[v]=scc_num;
            if(sta2[tp2--]==u) break;
        }
    }

}
void ini()
{
    memset(head,-1,sizeof(head));
    memset(col,0,sizeof(col));
    memset(gro,0,sizeof(gro));
    tp1=tp2=scc_num=0;
}
int main()
{
    while(~scanf("%d%d",&n,&m))
    {
        ini();
        for(int i=1;i<=m;i++)
        {
            int u,v;
            scanf("%d%d",&u,&v);
            es[i].u=u;
            es[i].v=v;
            es[i].next=head[u];
            head[u]=i;
        }

        for(int i=1;i<=n;i++) if(!dfn[i]) garbow(i,1);

        for(int i=1;i<=m;i++)
        {
            int u=es[i].u,v=es[i].v;
            if(gro[u]==gro[v])  continue;
            col[gro[u]]=1;
        }

        int cnt=0,t;
        for(int i=1;i<=scc_num;i++) if(col[i]==0) cnt++,t=i;
        if(cnt!=1) puts("0");
        else
        {
            int ans=0;
            for(int i=1;i<=n;i++) if(gro[i]==t) ans++;
            printf("%d\n",ans);
        }
    }
    return 0;
}
时间: 2024-10-13 11:35:56

POJ 2186 Popular Cows(初学强连通)的相关文章

poj 2186 Popular Cows 【强连通】

题目:poj 2186 Popular Cows 题意:n头牛,其中存在一些牛相互崇拜,具有传递性,问有多少头牛是被其他所有牛崇拜的. 分析:建立一个有向图,然后强连通缩点,之后求出度为0的点,假如存在多个,那么ans = 0,因为缩点之后如果x崇拜y,x也崇拜z,那么肯定y和z不能互相崇拜,不满足. 然后求出度为0的这个点缩点前环上有多少个点就ans AC代码: #include <cstdio> #include <vector> #include <iostream&g

POJ 2186 Popular Cows (强连通分量)

POJ 2186 Popular Cows 链接:http://poj.org/problem?id=2186 题意:每头奶牛都梦想着成为牧群中最受奶牛仰慕的奶牛.在牧群中,有N 头奶牛,1≤N≤10,000,给定M 对(1≤M≤50,000)有序对(A, B),表示A 仰慕B.由于仰慕关系具有传递性,也就是说,如果A 仰慕B,B 仰慕C,则A 也仰慕C,即使在给定的M 对关系中并没有(A, C).你的任务是计算牧群中受每头奶牛仰慕的奶牛数量. 思路:首先可以知道,在同一个强连通分量内的点一定互

poj 2186 Popular Cows 【强连通分量Tarjan算法 + 树问题】

题目地址:http://poj.org/problem?id=2186 Popular Cows Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 27496   Accepted: 11059 Description Every cow's dream is to become the most popular cow in the herd. In a herd of N (1 <= N <= 10,000) cows

Poj 2186 Popular Cows(Tarjan 强连通缩点)

传送门:Poj 2186 题意:给你n头牛,m种关系,A牛认为B牛是popular的,B牛认为C牛是popular的,则A也认为C是popular的,问最终有几头被所有牛认为是popular的牛 题解:强连通缩点基础题(虽然我Tarjan和缩点都是对的,但是最终讨论判断的时候写垮了(写了3天....还不是你懒!!!!过年划水这么多天缩点后找出度为零的点个数.然后讨论是否有这样子的点,如果没有则全都是(整个都是强连通图),如果只有一个,那么那个强连通分量所含的牛的个数就是所求解,如果有多个那么都是

POJ 2186 Popular Cows(强连通)

Popular Cows Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 30999   Accepted: 12580 Description Every cow's dream is to become the most popular cow in the herd. In a herd of N (1 <= N <= 10,000) cows, you are given up to M (1 <= M &

POJ 2186 Popular Cows --强连通分量

题意:给定一个有向图,问有多少个点由任意顶点出发都能达到. 分析:首先,在一个有向无环图中,能被所有点达到点,出度一定是0. 先求出所有的强连通分支,然后把每个强连通分支收缩成一个点,重新建图,这样,这个有向图就变成了一个有向无环图. 在这个新的图中,只需知道出度为0的点有几个即可. 如果出度为0的点超过1个,则输出0:否则输出出度为0的点所代表的那个强连通分支的分量数即可. 用Tarjan求强连通分量 代码: #include <iostream> #include <cstdio&g

强连通分量分解 Kosaraju算法 (poj 2186 Popular Cows)

poj 2186 Popular Cows 题意: 有N头牛, 给出M对关系, 如(1,2)代表1欢迎2, 关系是单向的且可以传递, 即1欢迎2不代表2欢迎1, 但是如果2也欢迎3那么1也欢迎3. 求被所有牛都欢迎的牛的数量. 限制: 1 <= N <= 10000 1 <= M <= 50000 思路: Kosaraju算法, 看缩点后拓扑序的终点有多少头牛, 且要判断是不是所有强连通分量都连向它. Kosaraju算法,分拆完连通分量后,也完成了拓扑序. /*poj 2186

POJ 2186 Popular Cows -- tarjan 缩点

链接: POJ 2186 Popular Cows 题意: 每一头牛都希望在牛群里面备受瞩目,在一个牛群中有N头牛(1<=N<=10000),你被给予M(1<=M<=50000)个关系对,形式如(A,B),这意味着A牛认为B牛比它更受欢迎,由于这种欢迎度是满足传递性的,那么若是A牛认为B牛更受欢迎,B牛认为C牛更受欢迎,那么A牛也会认为C牛更受欢迎.你的任务是计算出被所有牛受欢迎的牛的个数. 输入: 第一行两个整数 N 和 M 第2 到 M + 1 行,两个分开的数 A,B,意味着

Poj 2186 Popular Cows (Tarjan)

题目链接: Poj 2186 Popular Cows 题目描述: 有n只牛,牛之间存在一些关系,比如a认为b很受欢迎,b认为c很受欢迎,这样呢,a也会认为c很受欢迎,问根据给出的关系,有多少头牛被其他所有的牛都认为是受欢迎的? 解题思路: 对于一个有向无环图来说,其中有且仅有一个点出度为零,那么这个特殊的点,可以由其他任何点到达.那么接下来我们直接对所给的图进行强连通分量划分,然后把每个强连通分量看做一个点,判定出度为零的点有几个,如果有一个就输出这个点对应的强连通分量含有的节点个数,否则为零

POJ 2186 Popular Cows (强联通)

id=2186">http://poj.org/problem? id=2186 Popular Cows Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 23819   Accepted: 9767 Description Every cow's dream is to become the most popular cow in the herd. In a herd of N (1 <= N <=