zoukankan      html  css  js  c++  java
  • 拓扑排序 (DFS和BFS及判断是否有环)

    一、什么是拓扑排序?

    在图论中,拓扑排序(Topological Sorting)是一个有向无环图(DAG, Directed Acyclic Graph)的所有顶点的线性序列。且该序列必须满足下面两个条件:

    • 每个顶点出现且只出现一次。
    • 若存在一条从顶点 A 到顶点 B 的路径,那么在序列中顶点 A 出现在顶点 B 的前面。

    有向无环图(DAG)才有拓扑排序,非DAG图没有拓扑排序一说。

    拓扑排序其实还是挺奇妙的,就是解决谁先谁后的问题

    例如,下面这个图:

    它是一个 DAG 图,那么如何写出它的拓扑排序呢?这里说一种比较常用的方法:
    1. 从 DAG 图中选择一个没有前驱(即入度为0)的顶点并输出。

    2. 从图中删除该顶点和所有以它为起点的有向边。

    3. 重复 1 和 2 直到当前的 DAG 图为空或当前图中不存在无前驱的顶点为止

    于是,得到拓扑排序后的结果是 { 1, 2, 4, 3, 5 }。

    通常,一个有向无环图可以有一个或多个拓扑排序序列。这是因为可能同时存在多个入度为0的结点,这时,先处理哪个都是可以的。

    二. 拓扑排序用来干什么?

    • 判断是否有环
    • 求DAG中的最长链

    拓扑排序有两种方式,就是bfs和dfs,一般书中介绍的大多数是bfs,大家就以为拓扑排序只有一种办法,其实是不对的。

    参考链接 :https://blog.csdn.net/weixin_43918531/article/details/86740991

    三、拓扑排序的bfs模板

    有一个明确的思路:每一个顶点都有入度和出度,入度为0说明没有指向他的,那么就从他开始往下找。把这个入度为0的push进队列(还要注意保存入度为0的点),同时把与这个点相连的所有点的入度-1,然后再看看有没有入度为0的,有的话继续push,循环上面的操作,直到没有入度为0的点。
    看一下上面的图,如果从序号1开始的话:1入度为0,push进队列,顶点2的入度-1,所以顶点2push进队列,3和5的入度-1,3push进队列,5push进队列,顶点3进队列后,顶点4入度-1,顶点4push进队列,所以输出结果就是1 2 3 5 4

    #include <bits/stdc++.h>
    
    using namespace std;
    const int INF = 0x3f3f3f3f;
    
    
    //本代码功能:以bfs输出一个有向无环图DAG的拓扑序
    const int N = 1010;
    vector<int> edge[N];    //邻接表
    int ind[N];             //入度数组
    queue<int> q;          //队列
    
    int n;          //n个结点
    int m;          //m条边
    int main() {
        //读入,建图
        cin >> n >> m;
        for (int i = 1; i <= m; i++) {
            int x, y;
            cin >> x >> y;
            edge[x].push_back(y);
            ind[y]++;//维护入度
        }
        //入度为零的放入队列
        for (int i = 1; i <= n; i++) if (!ind[i]) q.push(i);
    
        //广度优先搜索DAG,就是拓扑排序的模板
        while (!q.empty()) {
            int x = q.front();
            q.pop();
            //输出拓扑序
            cout << x << " ";
            for (int i = 0; i < edge[x].size(); i++) { //遍历所有出边
                int y = edge[x][i];                    //目标结点
                //对接点入度-1,抹去这条入边
                ind[y]--;
                //如果入度为0,则入队列,准备处理它
                if (!ind[y]) q.push(y);
            }
        }
        return 0;
    }
    /**
    测试数据
     5 4
    
     1 2
     2 3
     2 5
     3 4
    
    参考答案:
    1 2 5 3 4
     */
    
    

    四、拓扑排序的dfs模板

    DFS是从一个点不断往下递归,比如说从序号1往下递归,有箭头就一直往下进行,直到到了最后一个元素,就开始往栈里(当然也可以是vector之类的,只不过需要反向再输出)push元素。比如说上面的从序号1开始,到序号2,序号3,序号4,到尽头了,就把4push进栈中,3push进栈,这个时候由于5也是2的下一个元素,所以5push进栈中,2push进栈,1push进栈,然后输出就是1 2 5 3 4.
    当然这个递归的顺序是与你输入的顺序有关的,不过思路都是这样的,由起始点向下递归。

    #include <bits/stdc++.h>
    
    using namespace std;
    const int INF = 0x3f3f3f3f;
    
    //本代码功能:以dfs输出一个有向无环图DAG的拓扑序
    const int N = 1010;
    
    bool st[N];             //标识是不是已经使用过
    vector<int> edge[N];    //邻接表
    vector<int> res;        //拓扑序列
    
    /**
     * 功能:深度优先搜索,记录拓扑序
     * @param u
     */
    void dfs(int u) {
        //如果访问过了,则返回,不再重复访问
        if (st[u])return;
        //标识u结点已使用
        st[u] = true;
    
        //遍历每个出边,找到下一组结点
        for (int v:edge[u]) if (!st[v]) dfs(v);
    
        //这一层完毕才把它自己扔进去,最后扔等于最先输出,因为后面是倒序输出的
        res.push_back(u);
    }
    
    int n;          //n个结点
    int m;          //m条边
    int main() {
        //读入,建图
        cin >> n >> m;
        for (int i = 1; i <= m; i++) {
            int x, y;
            cin >> x >> y;
            edge[x].push_back(y);
        }
        //将所有结点进行深入优先搜索
        for (int i = 1; i <= n; i++) dfs(i);
    
        //输出,从后向前输出
        for (int i = res.size() - 1; i >= 0; i--)
            cout << res[i] << " ";
    }
    
    /**
    测试数据
     5 4
    
     1 2
     2 3
     2 5
     3 4
    
    参考答案:
    1 2 5 3 4
     */
    
    

    五、dfs判断是否有环

    #include <bits/stdc++.h>
    using namespace std;
    
    /**
     5 5
    
     1 2
     2 3
     2 5
     3 4
     3 1
    
    对比上个例子,添加了一条3->1的边,就成了有向有环图.
    本题,也就不能输出拓扑序了,因为有环图没有拓扑序,拓扑序是针对DAG的。可以判断是否有环。
     */
    
    //本代码功能:以dfs判断一个有向图SDG是否有环
    const int N = 1010;
    
    int st[N];          //标识是不是已经使用过
    vector<int> edge[N];//邻接表
    int n;              //n个结点
    int m;              //m个关系
    
    /**
     * 功能:深度优先搜索,判断以u开头的图中是否有环,有环:true,无环:false
      有向有环图dfs判断是否有环只需要把st[]的状态改一下,原本是两种状态,0和1,
      现在改成 0,1,-1
      0:代表未访问
     -1:代表访问完毕
      1:代表是这一阶段正在访问的(这一阶段指的是两个元素在同一个递归中)。
     */
    bool dfs(int u) {
        //标识u结点正在访问
        st[u] = 1;
    
        //遍历每个出边,找到下一组结点
        for (int v:edge[u]) {
            //如果遇到了正在访问的结点,那么说明有环
            if (st[v] == 1) return true;
            //如果v这个结点没有访问过,递归查找v结点是否在环中
            if (st[v] == 0 && dfs(v)) return true;
        }
    
        //标识u结点访问完毕
        st[u] = -1;
        return false;
    }
    
    int main() {
        //读入,建图
        cin >> n >> m;
        for (int i = 1; i <= m; i++) {
            int x, y;
            cin >> x >> y;
            edge[x].push_back(y);
        }
        //将所有未访问过的结点进行深入优先搜索判断是否有环
        for (int i = 1; i <= n; i++)
            if (!st[i] && dfs(i))//没有访问过,并且有环
                cout << "发现环!" << endl;
        return 0;
    }
    
    

    六、bfs判断是否有环

    #include <bits/stdc++.h>
    
    using namespace std;
    
    /**
     5 5
    
     1 2
     2 3
     2 5
     3 4
     3 1
    
    对比上个例子,添加了一条3->1的边,就成了有向有环图.
    本题,也就不能输出拓扑序了,因为有环图没有拓扑序,拓扑序是针对DAG的。可以判断是否有环。
     */
    
    //本代码功能:以bfs判断一个有向图SDG是否有环
    const int N = 1010;
    
    int st[N];          //标识是不是已经使用过
    vector<int> edge[N];//邻接表
    int ind[N];         //入度表
    int n;              //n个结点
    int m;              //m个关系
    queue<int> q;       //队列
    vector<int> ans;   //ans 为拓扑序列
    int main() {
        //读入,建图
        cin >> n >> m;
        for (int i = 1; i <= m; i++) {
            int x, y;
            cin >> x >> y;
            edge[x].push_back(y);
            ind[y]++;
        }
        for (int i = 1; i <= n; i++) if (ind[i] == 0) q.push(i);  //将入度为0的点入队列
    
        while (!q.empty()) {
            int p = q.front();
            q.pop(); // 选一个入度为0的点,出队列
            ans.push_back(p);
            for (int i = 0; i < edge[p].size(); i++) {
                int y = edge[p][i];
                ind[y]--;
                if (ind[y] == 0) q.push(y);
            }
        }
        if (ans.size() == n) {
            for (int i = 0; i < ans.size(); i++)
                printf("%d ", ans[i]);
            printf("
    ");
        } else printf("No Answer!
    ");   //  ans 中的长度与n不相等,就说明无拓扑序列
    
        return 0;
    }
    
    
  • 相关阅读:
    mysql 在orderby和limit混合使用时重复数据问题
    springboot启动类 注解
    redis RDB和AOF两种持久化的区别
    C#解析逻辑字符串【x>y&&a>b||p=r】
    删除例如联想笔记本系统隐藏分区
    通过贝叶斯算法实现自动识别类别
    将可执行exe文件注册成windows服务
    Windows10中的IIS10安装php manager和IIS URL Rewrite 2.0组件的方法
    添加钩子监听全局鼠标或键盘事件
    C# DateTime.Now和DateTime.UtcNow的区别
  • 原文地址:https://www.cnblogs.com/littlehb/p/15125824.html
Copyright © 2011-2022 走看看