hdu 5154 拓扑排序

Posted 欧鹏

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了hdu 5154 拓扑排序相关的知识,希望对你有一定的参考价值。

例题:hdu 5154  链接  http://acm.hdu.edu.cn/showproblem.php?pid=5154

题目意思是第一行先给出n和m表示有n件事,m个关系,接下来输入m行,每行有a,b两个数,他们之间的关系就是第a件事要在

第b件事之前做完,然后问你可不可以在符合这m个关系的情况下把这n件事情做完,可以就输出YES,否则输出NO。

现在先假设a b之间的关系用a指向b来表示,即a-->b。

题目里给的两个样例有点简单,来个稍微复杂一点的好吧。

第一次搞这个,不太会,见谅。看上面这个图,然后我们可以先得到这些数据:

6 7

1 5

1 3

2 5

3 4

3 5

3 6

5 6

入度:就理解为一个点被箭头指的次数(作为终点的次数),上图中 点1的入度为0,点5的入度为3,点3的入度为1。

出度:反一下,箭头出去的次数(作为起点的次数),点1的出度为2,点5的为1。(出度在这题用不到)

我们程序就是先找到入度为0(一开始有点1和点2)的点,然后把这个点标记为走过,再把以这个点为起点的边删掉,

我们先选点1,然后就下面酱紫了。

 

然后入度为0的点就有点2和点3了,随便选择一个,假设3好吧,然后:

 

一直这样下去

然后就没有然后了,把点4删了就完了。

以上是符合条件输出YES的情况,假设图是这样的

 那么1和3就构成了一个环,入度为0的点只有2,然后删去2和以2位起点的边:


然后就没有入度为0的边了,然后程序结束,可是所有的点还没有走完(没有做完所有的事情),输出NO。

接下来就是怎么写的问题了。看代码吧

用vector的:

#include<iostream>
#include<algorithm>
#include<cstring>
#include<vector>
using namespace std;
vector<int>v[105];
int n,m,k,t;
int num[105],vis[105];//num数组记录入度数目,vis数组看这个点是否访问过 
void init()
{
    for(int i=1;i<=n;i++)
    v[i].clear();
    memset(num,0,sizeof(num));
    memset(vis,0,sizeof(vis));        
}
int main()
{
    while(scanf("%d%d",&n,&m)!=EOF)
    {
        init();
        for(int i=1;i<=m;i++)
        {
            int a,b;
            scanf("%d%d",&a,&b);
            num[b]++; //点b入度 +1 
            v[a].push_back(b);//a-->b 
        }
        int flag=0;
        for(int i=1;i<=n;i++)//最多n次循环 
        {
            for(int j=1;j<=n;j++)//找1到n入度为0的点 
            {
                if(vis[j])
                continue;
                if(num[j]==0)
                {
                    vis[j]=1;//标记这个点走过 
                    for(int k=0;k<v[j].size();k++)
                    {
                        if(num[v[j][k]])
                        num[v[j][k]]--;//删边 
                    }
                }
            }
        }
        for(int i=1;i<=n;i++)
        {
            if(!vis[i])//看是不是所有事都做完了 
            flag=1;
        }
        if(flag)
        printf("NO\\n");
        else
        printf("YES\\n");
    }
    return 0;
 } 

再用队列和链式前向星优化一下:

#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
int n,m,k,t;
struct node{
    int v,next;
}str[10005];
int head[105],vis[105],in[105],cnt;
queue<int>q;
void add(int u,int v)
{
    str[++cnt].v=v;
    str[cnt].next=head[u];
    head[u]=cnt;
}
void init()
{
    memset(vis,0,sizeof(vis));
    memset(head,-1,sizeof(head));
    memset(in,0,sizeof(in));
    cnt=0;
}
int main()
{
    while(scanf("%d%d",&n,&m)!=EOF)
    {
        init();
        for(int i=1;i<=m;i++)
        {
            int a,b;
            scanf("%d%d",&a,&b);
            add(a,b);
            in[b]++;
        }
        while(!q.empty())
        q.pop();
        for(int i=1;i<=n;i++)
        {
            if(!in[i])
            q.push(i);
        }
        while(!q.empty())
        {
            int u=q.front();
            q.pop();
            vis[u]=1;
            for(int i=head[u];i!=-1;i=str[i].next)
            {
                int v=str[i].v;
                if(in[v])
                in[v]--;
                if(in[v]==0&&!vis[v])
                q.push(v);
            }
        }
        int flag=1;
        for(int i=1;i<=n;i++)
        {
            if(!vis[i])
            {
                flag=0;
                break;
            }
        }
        if(flag)
        printf("YES\\n");
        else
        printf("NO\\n");
    }
    return 0;
}

 

以上是关于hdu 5154 拓扑排序的主要内容,如果未能解决你的问题,请参考以下文章

hdu 1285 确定比赛名次(拓扑排序)

HDU 5638 拓扑排序+优先队列

HDU[1285]确定比赛名次 拓扑排序

hdu1285 拓扑排序+优先队列

HDU-4857-逃生-反向拓扑排序+优先队列

hdu2647(拓扑排序)