图的拓扑排序(入门篇)

news2025/1/19 14:17:13

文章目录

    • 拓扑排序
    • 满足拓扑排序的前提:图中没有环
    • 拓扑排序的实现
    • 拓扑排序测试

拓扑排序

首先要说明一点:拓扑排序是针对图这种数据结构的特有排序。

百度百科对拓扑排序是这样定义的:

image-20221129171612918

上面的解释不是特别好懂,学过离散数学才知道偏序和全序的概念,这里我就给个通俗一点的理解:访问图的顶点时,保证每次访问的顶点前面没有别的顶点(入度为0),即访问的顶点只作为弧的弧头。

例如:

image-20221129200214048

满足拓扑排序的前提:图中没有环

如果出现了环,那么就会出现一种情况:在环中一直寻找入度为0的顶点,只有找到了,才能打破没有顶点入度为0的循环,但现在缺的就是环中入度为0的一个顶点。这就造成了死循环。如下:

image-20221129203100750

无论是从哪个路径入环,都会造成死循环。

拓扑排序的实现

访问顶点之前得先知道各个顶点的入度才行,因此得先遍历邻接矩阵,累计每个顶点的入度。并在访问完一个顶点之后,更新访问顶点的邻接点入度,也就是各自的入度自减1。这样循环检查,循环访问,直到所有的顶点被访问完就OK了。下面是用邻接矩阵来存储图的拓扑排序的代码实现。如果不是特别懂的话可以去看看我之前的博客(传送门),里面有详细讲解图的基本构造。

下面是图的一些基本构造和功能:

#include <iostream>
#include <vector>
#include <map>
#include <string>
#include <cstring>
using namespace std;
      顶点的数据类型 权值类型 权值的最大值默认为INT32_MAX,即∞    默认为无向图
template <class V, class W, W MAX_W = INT32_MAX, bool Direction = false>
class Graph//图的类框架
{
public:
    typedef Graph<V, W, MAX_W, Direction> Self;
    Graph() = default;
    Graph(const V *vertexs, int n)//顶点数组与个数传参
    {
        _vertexs.reserve(n);//初始化顶点数组与下标集
        for (int i = 0; i < n; ++i)
        {
            _vertexs.push_back(vertexs[i]);
            _indexMap[vertexs[i]] = i;
        }
        _matrix.resize(n);
        for (int i = 0; i < n; ++i)
        {
            _matrix[i].resize(n, MAX_W);//初始化邻接矩阵,每个元素默认赋值为∞
            _matrix[i][i]=W();//对角线上的权值为0
        }
    }
    size_t GetVertexIndex(const V& x)//获得顶点在顶点数组中的下标
    {
        auto it = _indexMap.find(x);
        if (it != _indexMap.end())//找到的话返回下标
        {
            return it->second;
        }
        else
        {
            throw invalid_argument("顶点不存在");//找不到的话就抛异常
            return -1;
        }
    }
    //添加关系(边)
    void _AddEdge(const size_t srci, const size_t dsti, const W& wight)
    {
        _matrix[srci][dsti] = wight;//两个顶点之间的权值进行赋值
        if (Direction == false)     //是无向图的话就在矩阵的对应位置也对边的权值进行赋值
        {
            _matrix[dsti][srci] = wight;
        }
    }
    //       参数:  起始顶点     终端顶点          权值
    void AddEdge(const V& src, const V& dst, const W& wight)
    {
        int srci = GetVertexIndex(src);//拿到起始顶点映射的下标
        int dsti = GetVertexIndex(dst);//拿到终端顶点映射的下标
        _AddEdge(srci, dsti, wight);   //嵌套一下用下标的方式添加边,方便在之后处理矩阵的特定情境通过下标加边。
    }
private:
    vector<V> _vertexs;        //顶点
    map<V, int> _indexMap;     //顶点对应的下标
    vector<vector<W>> _matrix; //矩阵
};

拓扑排序的代码:

//找所有顶点的入度
void FindInDegree(vector<int>& indegree) //indegree存的是所有顶点的入度
{
    size_t n = _vertexs.size(); //n为顶点的个数
    indegree.resize(n, 0); //入度初始化为0
    for (size_t i = 0; i < n; ++i) //对n个顶点进行各自的入度累计
    {
        for (size_t j = 0; j < n; ++j)
        {
            if (_matrix[i][j] != MAX_W && _matrix[i][j] != W())//有顶点i指向顶点j,顶点j的入度就自加1。
            {
                indegree[j]++;
            }
        }
    }
}
//拓扑排序         输出型参数 拓扑序列放在topo中
void TopologicalSort(vector<int>& topo)
{
    size_t n = _vertexs.size();//n为顶点的个数
    topo.resize(n, 0); //给topo开辟n个空间
    int index = 0;     //拓扑序列的实时下标,初始为0
    stack<int> st;        //存放入度为零的顶点
    vector<int> indegree; //所有顶点的初始入度
    vector<bool> visited(n, false); //标记映射的顶点是否被访问过
    FindInDegree(indegree);//查找所有顶点的入度
    for (size_t i = 0; i < n; ++i) // n个节点n次循环
    {
        for (size_t j = 0; j < n; ++j) //找未被访问且入度为零的顶点,将本次循环所有入度为0的顶点入栈
        {
            if (visited[j] == false && indegree[j] == 0)
            {
                st.push(j);
                visited[j] = true;//只要进栈,就意味着一定会被访问,直接先标记为已访问,避免重复检查
            }
        }
        int v = st.top();//取栈顶的入度为0的顶点下标
        topo[index] = v;//将该顶点下标存在拓扑序列中
        st.pop();//将该顶点出栈
        for (size_t j = 0; j < n; ++j)//已访问的顶点的所有未被访问的
        {
            if (visited[j] == false && _matrix[v][j] != MAX_W && _matrix[v][j] != W())
            {
                indegree[j]--; //更新已访问的顶点的邻接点的入度
            }
        }
        index++;//更新拓扑序列的实时下标
    }
}

拓扑排序测试

void TestTopologicalSort()
{
    vector<int> topo;
    const char* str="abcdef";
    Graph<char,int,INT32_MAX,true> g(str,strlen(str));
    g.AddEdge('a','b',1);
    g.AddEdge('a','d',1);
    g.AddEdge('a','c',1);
    g.AddEdge('d','e',1);
    g.AddEdge('f','d',1);
    g.AddEdge('f','e',1);
    g.AddEdge('c','b',1);
    g.AddEdge('c','e',1);
    g.TopologicalSort(topo);
    for(size_t i=0;i<topo.size();++i)
    {
        cout<<str[topo[i]]<<"["<<topo[i]<<"] ";
    }
    cout<<endl;
}

💻测试结果:

image-20221129214650832

图一摸一样,但是却和最开始我们得到的拓扑序列有所差别,这正好验证了拓扑序列在有些情况下并不唯一的情况。所以结果是没有错的,这是由于栈的先进后出特性导致的。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/47933.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

探索数字化节能降碳 广域铭岛助力电解铝行业碳达峰

近日&#xff0c;工信部等三部门联合印发《有色金属行业碳达峰实施方案》&#xff08;下称《方案》&#xff09;&#xff0c;要求确保2030年前有色金属行业实现碳达峰。 其中&#xff0c;针对电解铝行业&#xff0c;《方案》提出了优化冶炼产能规模、调整优化产业结构、强化技…

硬件接口和软件接口

文章目录硬件接口IDESCSISATA光纤通道游戏设备RAID卡USBMD设备MP3视频音频软件接口Java里的接口面向对象的接口聊聊软件接口1. 什么是接口2. 诞生3. 早期&#xff08;1950-1970&#xff09;4. 快速发展&#xff08;1970-1990&#xff09;5. 多元化发展&#xff08;1990-2010&am…

双胶合透镜初始设计

双胶合透镜是光学系统中不可或缺的基本光学零件之一。对于一个新设计的光学系统&#xff0c;首先根据性能要求对其进行外形尺寸计算&#xff0c;然后就得开始对各光学零部件进行初级像差设计&#xff0c;求解每个零部件的 、C的分配值&#xff0c;最后根据对各个零部件的 、C要…

小白学编程(JS):随机生成验证码

这道例题来自《JavaScipt从入门到精通》(第三版)中的【例6.6】。 书中给出的代码如下&#xff1a; <body><div id"result"> 产生的验证码&#xff1a;</div><input type"button" name"Submit" class"go-wenbenkuan…

基于PHP+MySQL公积金在线办理系统的设计与实现

公积金在线办理系统具有很强的信息指导性特征,采用PHP开发公积金在线办理系统 给web带来了全新的动态效果,具有更加灵活和方便的交互性。让企业、个人更加方便地在网上开展公积金等工作。 住房公积金是国家机关&#xff0c;企事业单位等及其所在在职职工缴存的长期住房储金。住…

RK3588平台开发系列讲解(CAN篇)CAN FD 开发文档

芯片名称内核版本安卓版本RK3588Linux 5.10Android 12🚀返回专栏总目录 文章目录 一、驱动文件二、DTS 节点配置三、内核配置四、CAN FD 通信测试工具五、CAN FD 常用命令接口沉淀、分享、成长,让自己和他人都能有所收获!😄 📢本篇将介绍CAN的使用及调试手段。 一、驱…

ADC的数据读取问题

一、从补码说起 计算机是如何表示负数的呢&#xff1f;这要从补码说起。 在数学中&#xff0c;任意基数的负数都在最前面加上"−"符号&#xff08;负号&#xff09;来表示。 然而&#xff0c;在计算机硬件中&#xff0c;数字都以无符号的二进制形式表示&#xff0…

[附源码]Python计算机毕业设计Django蛋糕购物商城

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

复旦-华盛顿大学EMBA 二十年20人丨林劲:对自己多一些“标准”

复旦大学-华盛顿大学EMBA20周年校友系列访谈。      一流企业定标准、二流企业做品牌、三流企业做产品。这是被广为传颂的一句话&#xff0c;意思是说要想成为一流企业&#xff0c;就必须成为行业标准的制定者&#xff0c;或至少能够主导标准的制定。尤其当企业需要在国际市…

2022年NPDP新版教材知识集锦--【第四章节】(3)

【实体化设计阶段】(全部获取文末) 实体化设计阶段是从概念定义开始&#xff0c;根据技术和经济性要求&#xff0c;不断进行设计&#xff0c;直至达到可用于制造的详细设计阶段&#xff0c;从而实现可制造性。 3.1联合分析 联合分析(Conjointanalysis)是一种统计分析方法&am…

python中protobuf和json互相转换应用

在实际信息系统开发中&#xff0c;经常会用到各种各样的协议&#xff0c;网络协议常用的有http&#xff0c;tcp&#xff0c;udp等&#xff0c;传输数据格式协议有json&#xff0c;xml&#xff0c;TLV等。本节将给大家介绍一种节省带宽数据协议&#xff0c;谷歌的ProtoBuf协议&a…

使用 Webmin+bind9快速搭建私有DNS服务器

什么是DNS DNS是Domain name system的简称&#xff0c;有些地方也称为Domain name server DNS主要是用于将域名解析为IP地址的协议&#xff0c;有时候也用于将IP地址反向解析成域名&#xff0c;所以DNS可以实现双向解析。 DNS可以使用TCP和UDP的53端口&#xff0c;基本使用U…

HTML篇_二、HTML简介_HTML入门必修第一课

HTML篇_二、HTML简介 一、HTML的基本结构 1.1 HTML的基本结构及解析 基本结构 这里我们先放一段代码块来进行展示&#xff0c;感受一下来自HTML的魅力。然后下文再对这段代码块进行解析。 <!DOCTYPE html> <html><head><meta charset"utf-8&quo…

计算机组成原理习题课第三章-3(唐朔飞)

计算机组成原理习题课第三章-3&#xff08;唐朔飞&#xff09; ✨欢迎关注&#x1f5b1;点赞&#x1f380;收藏⭐留言✒ &#x1f52e;本文由京与旧铺原创&#xff0c;csdn首发&#xff01; &#x1f618;系列专栏&#xff1a;java学习 &#x1f4bb;首发时间&#xff1a;&…

JavaWeb

1、基础概念 静态web&#xff1a;html、css。给人看的数据始终不会发生改变。&#xff08;数据无法持久化&#xff0c;用户无法交互&#xff09; 动态web&#xff1a;①、淘宝、几乎所有网站&#xff1b; ②、给人看的数据始终会发生改变&#xff1b; ③、技术栈Servlet/JSP、…

可观测数据采集端的管控方案的简单对比

概述 当前&#xff0c;主流的日志采集产品除了SLS的ilogtail&#xff0c;还有Elastic Agent、Fluentd、Telegraf、Sysdig、Logkit、Loggie、Flume等。详细的对比结果见下表&#xff1a; 备注&#xff1a; 集群监控&#xff1a;表示工具可以查看管理采集端的运行状态、采集速…

iClient for MapboxGL对接WMS服务

作者&#xff1a;yx 文章目录前言一、获取WMS服务二、请求参数说明三、获取参数四、关键代码五、完整代码总结前言 咱们iClient官网Leaflet、OpenLayers、Classic均有对接WMS服务的示例&#xff0c;详情可以参考iClient官网示例https://iclient.supermap.io/。 但是许多小伙伴…

基础知识java

1.浅克隆和深克隆&#xff1f;深克隆的方法 浅克隆&#xff1a;对象的引用变量只会拷贝地址&#xff0c;不会新建一个对象 深克隆&#xff1a;对象的引用变量也会新建一个对象 实现方式&#xff1a; 浅克隆&#xff1a;实现cloneable接口的clone方法 深克隆&#xff1a;实现Ser…

《树莓派项目实战》第八节 使用光敏电阻传感器检测环境中是否有光照

目录 8.1 引脚介绍 8.2 工作原理 8.3 连接到树莓派 8.4 编写代码检测有无光照 在本节&#xff0c;我们将学习如何使用光敏电阻度传感器检测是否有光照&#xff0c;该项目设计到的材料有&#xff1a; 树莓派 * 1面包板 * 1杜邦线若干光敏电阻传感器 * 18.1 引脚介绍 从右到…

tictoc例子理解6-9

tictoc 6-9tictoc 6 自消息实现计时tictoc 7 节点等待时延随机&#xff0c;丢失包概率tictoc 8 两个节点分别定义两个类tictoc 9 保留原始包副本&#xff0c;从而不需要重新构建包tictoc 6 自消息实现计时 在前面的模型中&#xff0c;’ tic’和’ toc’立即将收到的消息发送回…