zoukankan      html  css  js  c++  java
  • 拓扑排序 topsort

    拓扑排序

      对一个有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序,是将G中所有顶点排成一个线性序列,使得图中任意一对顶点u和v,若边(u,v)∈E(G),则u在线性序列中出现在v之前。通常,这样的线性序列称为满足拓扑次序(Topological Order)的序列,简称拓扑序列。简单的说,由某个集合上的一个偏序得到该集合上的一个全序,这个操作称之为拓扑排序。

    AOV网

      在有些工程中,很多子工程(活动)必须在其它有关子工程完成之后才能开始,也就是说,一个子工程的开始是以它的所有前序子工程的结束为先决条件的,我们把这种顶点表示活动、边表示活动间先后关系的有向图称做顶点活动网(Activity On Vertex network),简称AOV网。

      由AOV网构造出拓扑序列的实际意义是:如果按照拓扑序列中的顶点次序,在开始每一项活动时,能够保证它的所有前驱活动都已完成,从而使整个工程顺序进行,不会出现冲突的情况。

    求拓扑序通常使用两种方法:(以下代码可通过UVa 10305)

    DFS 求拓扑序

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    
    using namespace std;
    
    int n, m, x, y;
    int q[200], t;
    
    bool e[200][200];
    
    short v[300];
    
    bool dfs(int u)
    {
        v[u] = -1;
        for (int i = 1; i <= n; ++i)
            if (e[u][i]) if (!v[i] < 0) return false;
                else if (!v[i] && !dfs(i)) return false;
        v[u] = 1;
        q[--t] = u;
        return true;
    }
    
    bool top_sort(void)
    {
        t = n + 1;
        memset(v, 0, sizeof(v));
        for (int i = 1; i <= n; ++i)
            if (!v[i]) if (!dfs(i)) return false;
        return true;
    }
    
    int main()
    {
        while (cin >> n >> m && n)
        {
            memset(e, false, sizeof(e));
            memset(ind, 0, sizeof(ind));
            for (int i = 1; i <= m; ++i)
            {
                scanf("%d%d", &x, &y);
                e[x][y] = true;
            }
            if (top_sort())
                for (int i = 1; i <= n; ++i)
                    printf("%d ", q[i]);
            else printf("-1");
            printf("
    ");
        }
    }
    

    BFS 求拓扑序

    #include <iostream>
    #include <cstdio>
    #include <cstring>
    
    using namespace std;
    
    int n, m, x, y;
    int q[200];
    int ind[200];
    
    bool e[200][200];
    
    void top_sort(void)
    {
        int h = 1, t = 0;
        for (int i = 1; i <= n; ++i)
            if (ind[i] == 0)
                q[++t] = i;
        while (h <= t)
        {
            int u = q[h++];
            for (int i = 1; i <= n; ++i)
                if (e[u][i])
                {
                    --ind[i];
                    if (ind[i] == 0)
                        q[++t] = i;
                }
        }
        for (int i = 1; i <= n; ++i)
            printf("%d ", q[i]);
            printf("
    ");
    }
    
    int main()
    {
        while (cin >> n >> m && n)
        {
            memset(e, false, sizeof(e));
            memset(ind, 0, sizeof(ind));
            for (int i = 1; i <= m; ++i)
            {
                scanf("%d%d", &x, &y);
                e[x][y] = true;
                ++ind[y];
            }
            top_sort();
        }
    }
    
  • 相关阅读:
    在MAC OS X系统上面安装mysql
    在项目中使用DSOFramer需要注意的一些地方
    记一次python安装PIL库所遇到的事
    Calendar.compareTo 比较时间的大小
    String 时间类型怎么进行比较大小?
    ElasticSearch--二、基本语法(创建索引,查询数据)
    ElasticSearch--一、使用场景以及对应软件配置安装
    Linux下nginx反向代理负载均衡几种方式以及配置
    node环境使用lowdb轻量数据库以及基本用法
    jQuery 日常笔记
  • 原文地址:https://www.cnblogs.com/yanyiming10243247/p/9666967.html
Copyright © 2011-2022 走看看