Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >拓扑排序及其实际应用

拓扑排序及其实际应用

作者头像
用户1168362
发布于 2018-01-05 07:54:58
发布于 2018-01-05 07:54:58
2.1K00
代码可运行
举报
文章被收录于专栏:.net core新时代.net core新时代
运行总次数:0
代码可运行

  最近在做实际项目中遇到了一个问题,如何判断一个层级结构的图是否存在循环引用。刚开始想到了方法是用递归进行判断,后来想到大学学过的拓扑排序可以解决该问题,于是翻了下数据结构这本书,阅读了园友的文章,根据自己的理解写下了这篇随笔。

阅读目录

  • 拓扑排序介绍
  • 问题引入及算法实现
  • 本章总结

回到顶部

拓扑排序介绍

  百度百科定义:

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

  上面的定义看完可能不知道是什么意思,举两个实际的例子就明白了。

1.大学课程排序

大学课程的学习是有先后顺序的,C语言是基础,数据结构依赖于C语言,其它课程也有类似依赖关系。这样的一个课程安排是怎么实现的呢?

  2.VS项目编译顺序

   假设VS中有三个项目A,B,C,它们的关系如下图。VS编译器是如何判断三个项目的编译顺序的呢?

回到顶部

问题引入及算法实现

   这次实际项目中碰到的问题可以归纳为控件联动选择,即常见的省份,城市,地区联动。为了实现通用的下拉连dog,设计了一套表结构,最终保存数据如下。

     看到这里也许你不明白这个和拓扑排序能扯上什么关系,假如省份下拉又依赖于地区下拉,那这样就会形成一个死循环。为了避免这样的情况需要在数据保存时,校验是否存在闭环。

     下面给出,解决上述问题的两种算法。

1.递归判断

     步骤如下

      (1)找当前节点的父级节点(也可以叫依赖的节点)  

  (2)父级节点不为为空且不等于当前节点自己,则寻找父级节点对应的父级节点

      (3)重复1,2。最终找到的节点=自己 ,则存在闭环,否则不存在

代码实现

   首先定义了一个类似的结构   

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    public class Node
    {
        /// <summary>
        /// 当前节点ID
        /// </summary>
        public int Key { get; set; }

        /// <summary>
        /// 父级节点ID
        /// </summary>
        public int? Parent { get; set; }
    }
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/// <summary>
    /// 递归判断是否存在循环引用
    /// </summary>
    public class RecursionSort
    {
        /// <summary>
        /// 递归判断是否存在循环引用
        /// </summary>
          public  static bool CheckRecursion(List<Node> list)
        {
            foreach (var node in list)
            {
                if (RecursionSort.CheckRecursion(node.Key,node, list))
                {
                    return true;
                }
            }
            return false;
        }

        /// <summary>
        /// 递归判断是否存在循环引用
        /// </summary>
        /// <param name="list"></param>
        /// <returns></returns>
        private static bool CheckRecursion(int key,Node curNode, List<Node> list)
        {
            if (curNode.Parent == key)
            {
                return true;
            }
            //寻找父级节点对应的父级节点信息
            if (curNode.Parent != null)
            {
                Node pNode = list.Where(e => e.Key == curNode.Parent).FirstOrDefault<Node>();
                return CheckRecursion(key,pNode, list);
            }
            return false;
        }
    }
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
        static void Main(string[] args)
        {
            //递归判断
            List<Node> list = new List<Node>();
            list.Add(new Node { Key=1,Parent=2});
            list.Add(new Node { Key = 2, Parent = 1 });
            list.Add(new Node { Key = 3, Parent = 2 });
            Console.WriteLine(RecursionSort.CheckRecursion(list));
            Console.Read();
        }

2.拓扑排序

   步骤如下

        (1) 从有向图中选择一个出度为0(即不依赖任何其它节点)的顶点并且输出它。     (2) 从图中删去该顶点,并且删去该顶点的所有边。         (3) 重复上述两步,直到剩余的图中没有出度为0的顶点。

      我们来看一下上面举的VS项目编译顺序列子按照上述算法的演示过程

     第一步选择 C节点

      第二步选择 B节点

       至此完成了整个排序C,B,A 即先编译C项目,再编译B项目,最后编译A项目

    代码实现如下

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    /// <summary>
    /// 拓扑节点类。
    /// </summary>
    public class TopologicNode<T>
    {
        /// <summary>
        /// 获取或设置节点的键值。
        /// </summary>
        public T Key { get; set; }

        /// <summary>
        /// 获取或设置依赖节点的键值列表。
        /// </summary>
        public List<T> Dependences { get; set; }
    }
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 /// <summary>
    /// 拓扑排序类。
    /// </summary>
    public class TopologicSort
    {
        /// <summary>
        /// 拓扑顺序。
        /// </summary>
        /// <typeparam name="TKey">节点的键值类型。</typeparam>
        /// <param name="list">一组节点。</param>
        /// <returns>拓扑序列。</returns>
        /// <exception cref="InvalidOperationException">如果存在双向引用或循环引用,则抛出该异常。</exception>
        public static List<T> OrderBy<T>(List<TopologicNode<T>> list)
        {
            if (list == null)
            {
                throw new ArgumentNullException("参数空异常");
            }
            List<T> listResult = new List<T>();
            while (list.Count > 0)
            {
                //查找依赖项为空的节点
                var item = list.FirstOrDefault(c => c.Dependences == null || c.Dependences.Count == 0);
                if (item != null)
                {
                    listResult.Add(item.Key);

                    //移除用过的节点,以及与其相关的依赖关系
                    list.Remove(item);
                    foreach (var otherNode in list)
                    {
                        if (otherNode.Dependences != null)
                        {
                            otherNode.Dependences.Remove(item.Key);
                        }
                    }
                }
                else if (list.Count > 0)
                {
                    //如果发现有向环,则抛出异常
                    throw new InvalidOperationException("存在循环引用");
                }
            }
            return listResult;
        }
    }
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 //拓扑排序
            //节点3依赖于2和1节点
            list.Add(new Node { Key = 3, Parent = 1 });

            List<TopologicNode<int>> listTopologicNode = new List<TopologicNode<int>>();
            //构建排序节点

            var group = (from p in list
                         group p by p.Key into g
                         select g);

            foreach (var g in group)
            {
                TopologicNode<int> node = new TopologicNode<int>();
                node.Key = g.Key;
                node.Dependences = new List<int>();
                foreach (Node value in g)
                {
                    if (value.Parent != null)
                    {
                        node.Dependences.Add(value.Parent.GetValueOrDefault());
                    }
                }
                listTopologicNode.Add(node);
            }

            try
            {
                List<int> result = TopologicSort.OrderBy<int>(listTopologicNode);
                result.ForEach(e => {
                    Console.WriteLine(e);
                });
            }
            catch (Exception ex)
            {
                Console.WriteLine(ex.Message);
            }

运行结果如下

回到顶部

本章总结

     本篇用到了Linq语法,如有不懂的可以到园里找找相关知识。后续我会专门写一篇关于Linq,函数委托的文章,敬请期待!第一篇写算法的随笔到此完成,拓扑排序的实际应用场景还有很多,最短路径等等。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2015-04-19 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
图Graph--拓扑排序(Topological Sorting)
一个项目往往会包含很多代码源文件。编译器在编译整个项目时,需按照依赖关系,依次编译每个源文件。比如,A.cpp依赖B.cpp,那在编译时,编译器需要先编译B.cpp,才能编译A.cpp。 编译器通过分析源文件或者编译配置文件(比如Makefile文件),来获取这种局部的依赖关系。那编译器又该如何通过源文件两两之间的局部依赖关系,确定一个全局的编译顺序呢?
Michael阿明
2021/02/20
5990
图Graph--拓扑排序(Topological Sorting)
拓扑排序原理与解题套路
其实最开始学习算法,听到拓扑排序这几个字我也是懵逼的,后来学着学着才慢慢知道是怎么一回事。关于 拓扑 这个词,在网上找到这么一段解释:
五分钟学算法
2019/09/03
4950
无回路有向图的拓扑排序
因公司业务需要,在表单中每个字段都会配置自动计算,但自动计算公式中会引用到其他字段中的值。所以希望可以根据计算公式,优先计算引用的公式。所以最终使用了无回路有向图的扩扑排序来实现。
兜兜毛毛
2019/10/23
9510
无回路有向图的拓扑排序
一种非大小排序(先后关系排序)—拓扑排序
在以前很多人可能听过拓扑排序,但可能认为它太难而不愿接触学习,也不清楚是排啥序的,然而拓扑排序实际很简单,生活中也很常用,面试笔试也会遇到,所以掌握拓扑排序已是必要的!
全菜工程师小辉
2019/09/16
1.4K0
一种非大小排序(先后关系排序)—拓扑排序
【JavaScript 算法】拓扑排序:有向无环图的应用
DFS方法通过递归遍历图,将访问过的节点存入栈中,最终从栈顶依次取出节点构建拓扑序列。
空白诗
2024/07/20
3150
【JavaScript 算法】拓扑排序:有向无环图的应用
从分手厨房看拓扑排序
分手厨房(Over Cooked!)是一款以高难度合作著称的游戏,在形形色色的厨房中,你需要和你的同伴一起克服重重难关,按照指定的顺序生产出美味佳肴,满足客人的味蕾。在游戏过程中,制作一道菜需要完成许多的步骤,以第一关中的寿司为例,需要蒸米饭、切鱼片、切黄瓜、然后用紫菜把他们包在一起,与此同时你还要兼顾洗掉脏盘子。不难看出,当有多个玩家参战的时候,这里有些工序是可以同时进行的(比如蒸米饭和切鱼片),但也有些工序是有顺序依赖的(比如只有一个案板,那么切鱼片和切黄瓜就不可能同时进行),那么,如何才能将所有的工序进行一个合理的排序,来保证其正常运作呢?
程序员小猿
2021/01/19
5610
从分手厨房看拓扑排序
拓扑排序,YYDS!
很多读者留言说要看「图」相关的算法,那就满足大家,结合算法题把图相关的技巧给大家过一遍。
labuladong
2021/09/23
5980
拓扑排序
拓扑排序:如果图中从v到w有有一条有向路径,则v一定要排在w之前。满足此条件的顶点序列称为一个拓扑序。获得拓扑序的过程就是拓扑排序。
AI那点小事
2020/04/20
6480
拓扑排序
八十六、从拓扑排序探究有向图
关于排序,其实还有很多,比如常见的希尔排序,桶排序,计数排序和基数排序,由于要过渡到数据结构有向图,因此需要了解拓扑排序和邻接矩阵概念。
润森
2022/08/17
4660
八十六、从拓扑排序探究有向图
算法练习(18)-图的拓扑排序
就这个示例而言,显然正确的编译顺序是:5->4->3->2->1 或 4->5->3->2->1 (注:4与5之间没有相互依赖,谁先谁后都可以)
菩提树下的杨过
2021/11/10
4000
【拓扑排序】图论拓扑排序入门
在有向图中,以某个节点为起始节点,从该点出发,每一步沿着图中的一条有向边行走。如果到达的节点是终点(即它没有连出的有向边),则停止。
宫水三叶的刷题日记
2021/12/15
1.5K0
【拓扑排序】图论拓扑排序入门
算法细节系列(17):有向环检测&&拓扑排序
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/u014688145/article/details/71719530
用户1147447
2019/05/26
7250
环检测算法及拓扑排序(修订版)
首先,我说将后序遍历结果进行反转就是拓扑排序的结果,有的读者说他看到的很多解法直接使用后序遍历,并没有进行反转,本文新增了对此的解释。
labuladong
2022/03/30
1.3K0
环检测算法及拓扑排序(修订版)
AOV网与拓扑排序
图解演示拓扑排序的过程 不是AOV网的情况 由下可知,当前图中存在回路,不是AOV网 拓扑排序的数据结构 编程流程 拓扑排序算法伪代码 完整代码 #include<iostream> using namespace std; #include<stack> #define MAX 10//最大顶点数为10 //边表结构体 typedef struct EdgeNode { int agjvex;//邻接点域,存储顶点对应的下标 //int weight;//用来存储权值,对于非网图这
大忽悠爱学习
2021/04/01
3840
AOV网与拓扑排序
揭开「拓扑排序」的神秘面纱
上篇文章 的投票让我有点无奈,大家是不是都商量好了?那就。。anyway 这篇先来拓扑排序~
JavaFish
2020/08/24
4940
揭开「拓扑排序」的神秘面纱
BFS:解决拓扑排序问题
要知道什么拓扑排序我们首先要知道什么是有向无环图,有向无环图我们看名字其实就很容易理解,有向就是有方向,无环就是没有环形结构,这里我们展示一下有向无环图和有向有环图:
用户11305458
2024/10/09
1630
BFS:解决拓扑排序问题
Android 启动优化(二) - 拓扑排序的原理以及解题思路
春节之前,更新了一篇博客 Android 启动优化(一) - 有向无环图,反响还不错,今天,让我们一起来看一下,怎样用代码实现有向无环图。
程序员徐公
2021/03/15
6490
力扣207——课程表
在选修某些课程之前需要一些先修课程。例如,想要学习课程 0 ,你需要先完成课程 1 ,我们用一个匹配来表示他们: [0,1]
健程之道
2020/02/12
5230
力扣207——课程表
拓扑排序
在图论中,拓扑排序(Topological Sorting)是一个有向无环图(DAG, Directed Acyclic Graph)的所有顶点的线性序列。且该序列必须满足下面两个条件:
风骨散人Chiam
2020/10/28
6310
浅谈什么是图拓扑排序
  在工程实践中,一个工程项目往往由若干个子项目组成。这些子项目间往往有两种关系:   (1) 先后关系,即必须在某个项完成后才能开始实施另一个子项目。   (2) 子项目间无关系,即两个子项目可以同时进行,互不影响。
五分钟学算法
2019/05/14
2.4K0
浅谈什么是图拓扑排序
相关推荐
图Graph--拓扑排序(Topological Sorting)
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档