UVa 10305 - Ordering Tasks ( 拓扑排序, DFS, DAG )

题意

假设有n个变量,还有m个二元组(u, v),分别表示变量u小于v。那么,所有变量从小大排列起来应该是什么样子的呢?例如,有4个变量a, b, c, d,若已知a < b,c < b,d < c,则这4个变量的排序可能是a < d < c < b。尽管还有其他可能(如d < a < c < b),你只需找出其中一个即可

思路

无环图DAG才存在拓扑排序, 用mrk[]标记三种状态: -1(正在访问), 0(未访问), 1(访问过且已返回)
如果递归dfs遇到-1说明是有环图, 则不存在拓扑排序

AC代码

#include <iostream>
#include <cstdio>
#include <cstring>

using namespace std;

const int maxn = 100 + 50;
int dag[maxn][maxn];
int t[maxn], mrk[maxn];
int n, m, tp;

bool dfs( int x ){
    mrk[x] = -1;
    for( int i = 1; i <= n; i++ ){
        if( dag[x][i] ){
            if( mrk[i] == -1 )   return false;
            else if( !mrk[i] && !dfs(i) )  return false;
        }
    }
    mrk[x] = 1;
    t[--tp] = x;
    return true;
}

bool topo()
{
    tp = n;
    for( int i = 1; i <= n; i++ )
        if( !mrk[i] && !dfs(i) )
            return false;
    return true;
}

int main()
{
    int a, b;
    while( scanf("%d%d",&n, &m) == 2 && n ){
        memset(dag, 0, sizeof(dag));
        memset(mrk, 0, sizeof(mrk));
        while( m-- ){
            scanf("%d%d",&a,&b);
            dag[a][b] = 1;
        }
        topo();
        for( int i = 0; i < n; i++ ){
            if( i != 0 )    putchar(' ');
            printf("%d",t[i]);
        }
        putchar('\n');
    }
    return 0;
}
posted @ 2018-03-06 20:13  JinxiSui  阅读(137)  评论(0编辑  收藏  举报