拓扑排序

拓扑排序

简介

拓扑排序是将偏序的数据线性化的一种排序方法。复习下偏序和全序的概念:

全序关系是偏序关系的一个子集。

全序是集合内任何一对元素都是可比较的,比如数轴上的点都具有一个线性的数值,因此根据数值就可以进行比较。

偏序是集合内不是所有元素都是可以比较的,比如平面内的点由横坐标和纵坐标组成,是不可直接比较大小的。这是因为横坐标和纵坐标是两个维度,在每个维度内都可以用数值比较,但是维度之间不可量化比较(就像学习成绩和身体素质之间无法量化比较)。当然偏序是个数学概念,未必是多维度引发的不可比较,只需满足以下关系即满足偏序关系:

设 P 是集合,P 上的二元关系”≤”满足以下三个条件,则称”≤”是 P 上的偏序关系(或部分序关系):

(1)自反性:a≤a,∀a∈P;

(2) 反对称性:∀a,b∈P,若 a≤b 且b≤a,则 a=b;

(3) 传递性:∀a,b,c∈P,若 a≤b 且b≤c,则 a≤c;

注意这里的”≤”是一个自定义的二元运算符,而不是通常的线性运算的大小关系。

理解了偏序关系之后,拓扑排序就是将偏序关系线性化。举一个具体场景,在有向无环图中,”节点 A 是否可由节点 B 到达”即是一种偏序关系。在该有向无环图中节点,在不移动的情况下可以到达自身,因此满足自反性。且在有向无环图中不存在环,若 A 可到达 B 且 B可到达A,则A,B 必是相同节点,因此满足反对称性。当节点 A 可以到达节点 B,且节点 B 可以到达节点C 时,节点 A 也可以到达节点 C,因此满足传递性。

拓扑排序

在该场景下,拓扑排序即是将有向无环图中所有节点按照”节点 A 是否可由节点 B 到达”来进行线性化排序。如上图所示,对它进行拓扑排序可以使用深度优先算法完成,深度优先算法可以复习课件,其过程大概如下。

  1. 选取顶点 s(一般选取入度比较小的节点更合适,比如上图的节点 1),标记状态
  2. 若 s 有未被访问的邻居,则选择邻居标记状态继续访问,否则返回
  3. 如果一次深度优先搜索还有节点未被访问,则重复步骤 1,直到所有节点被访问到

这种方法可以将满足偏序关系的有向无环图线性化为🎧一种排序结果:1,2,4,3,5。当然对于更复杂的有向无环图可能有多种合法的排序结果。

实现

实现代码如下:

//
// Created by lenovo on 2022/5/1.

//
#include

#include "iostream"
#include "vector"
#include "fstream"
using namespace std;

typedef struct EdgeNode
{
    int index;                  //邻接点
    struct EdgeNode *next;      //链表,指向下一个邻接点
}EdgeNode;

typedef struct PointNode        //顶点表节点
{
    int in;                     //顶点入度
    int data;                   //顶点信息
    EdgeNode* firstEdge;        //边表头指针
    PointNode(){in=NULL;data= NULL;firstEdge= nullptr;}
}PointNode;

typedef struct Graph{
    int NumPoint,NumEdge;
    PointNode* arr;
}Graph;

void read_file(Graph* G){           //构造图Graph
    ifstream inputData;
    string input_file_path="..\\input.txt";     //示例输入在同级目录input.txt
    inputData.open(input_file_path, ios::in);
    string line;
    int tmp=0;
    int i=0;
    EdgeNode *e;
    while (inputData>>line){
        int bracketPos = line.find(',');
        int from = stoi(line.substr(0, bracketPos));
        int to= stoi(line.substr(bracketPos+1,line.size()-bracketPos));
        G->NumEdge++;                               //增加边
        if(from!=tmp){tmp=from;G->NumPoint++;}      //增加新的节点

        /*保存边信息*/
        e = (EdgeNode*)malloc(sizeof(EdgeNode));
        e->index = to;

        e->next = G->arr[from].firstEdge;
        G->arr[from].firstEdge = e;
        G->arr[to].in++;
    }
    inputData.close();

}

/*获取输入的边的数量*/
int read_num(){
    ifstream inputData;
    string input_file_path="..\\input.txt";
    inputData.open(input_file_path, ios::in);
    string line;
    int tmp=0;
    while (inputData>>line){
       tmp++;
    }
    inputData.close();
    return tmp;
}

int ToupuSort(Graph* G){

    EdgeNode* edge;
    int next;
    vector stack;
    int count=0;
    for(int i=0;iNumPoint;++i){             //栈stack存储入度为0的节点,需排除0节点
        if(G->arr[i].in==0)
            stack.push_back(i);
    }
    int out;
    while (stack[0]!=stack.back()){                       //当栈为非空
        out=stack.back();
        cout<arr[out].firstEdge;edge;edge=edge->next){           //更新邻接点的入度,-1
            next=edge->index;
            if(!(--G->arr[next].in))       //邻接节点入度原为1,则入栈
                stack.push_back(next);

        }
    }

    if(countNumPoint){              //有环
        return 0;
    }else
        return 1;          //无环,且全部输出

}

int main(){
    Graph G;
    PointNode arry[read_num()];
    G.arr=arry;
    read_file(&G);      //构建图
    ToupuSort(&G);      //输出拓扑排序
    return 0;
}

示例输入:

1,2
1,4
2,4
2,3
3,5
4,3
4,5

示例输出:

1&#xFF0C;2&#xFF0C;4&#xFF0C;3&#xFF0C;5

Original: https://www.cnblogs.com/world-explorer/p/16213100.html
Author: O_fly_O
Title: 拓扑排序

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/607383/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • Linux -查找功能

    Linux下有很多用于查询的命令,持续更新ing find命令 find用于查找指定目录下的文件 语法: find 【查找的位置】(选项) (参数) [filename] 常用选项…

    Linux 2023年5月27日
    0120
  • 修改内核中的只读区内容

    研究到 apparmor 内核源码,其中涉及到只读变量 __lsm_ro_after_init,研究怎么修改只读区实现部分功能,这里记录一下。 思路上很简单,修改的时候禁用写保护,…

    Linux 2023年6月13日
    079
  • 安卓投屏助手(ARDC)最新版

    近几年安卓多屏协同非常火爆,以华为小米为首的各大手机厂商都推出了各自的多屏协同软件,打破手机、平板和VR等安卓设备与电脑的边界,通过多屏融合提高办公的生产力。国内安卓投屏软件有To…

    Linux 2023年6月7日
    0317
  • 【MQTT】cJSON协议的安装和使用

    cJSON的使用 * – cJSON的简介 – + JSON 名称/值对 + JSON 数字 + JSON 对象 + JSON 数组 – cJS…

    Linux 2023年6月13日
    0111
  • 【河北科技大学数据结构课设】校园导航问题

    文档到我的资源下载 点击这里进入我的资源下载 1. 简单介绍 2. 代码 #include #include #include using namespace std; /*测试使…

    Linux 2023年6月8日
    0115
  • Redis6 源码调式

    Redis6 源码调式 安装Cygwin 1、下载安装Cygwin 去Cygwin的官方网站http://www.cygwin.com/ window 64位请选择 setup-x…

    Linux 2023年5月28日
    0124
  • Android recovery支持adb shell

    Android recovery支持adb shell 近期开发过程注意到recovery不支持adb shell。为了便于调试方便,决定添加此功能。 刚開始我们採用的是user版…

    Linux 2023年5月28日
    099
  • docker 安装nextcloud+onlyoffice+mysql

    环境 类目 版本 备注 操作系统 centos 7 64位 Docker 最新版 Docker compose 最新版 暂时没有用上可以不安装 nextcloud 最新版 only…

    Linux 2023年5月27日
    094
  • Redis 基础

    Redis 基础 Redis 定位 – 特性 关系型数据库 特性 非关系型数据库 特性 Redis 特性 Redis 安装 – 启动 – 使用 …

    Linux 2023年6月13日
    0139
  • 在Ubuntu20.04上安装Kubernetes-Kubeadm和Minikube

    镜像下载、域名解析、时间同步请点击阿里云开源镜像站 在本文中,我们将了解如何在 Ubuntu 20.04 上安装 Kubernetes。在过去的几年里,容器化为开发人员提供了很大的…

    Linux 2023年5月27日
    0107
  • 浅谈DDD中的聚合

    DDD分为战略部分跟战术部分,相信大家都认同DDD的核心在战略而非战术。而战略方面的核心我认为在业务建模,领域划分、统一语言等都在为业务建模服务。 为什么业务建模重要? 以前的开发…

    Linux 2023年6月8日
    096
  • 附034.Kubernetes_v1.21.0高可用部署架构二

    kubeadm介绍 kubeadm概述 kubeadm功能 本方案描述 部署规划 节点规划 主机名配置 变量准备 互信配置 环境初始化 部署高可用组件 HAProxy安装 Keep…

    Linux 2023年6月13日
    0231
  • Linux系统僵尸进程详解

    大安好,我是良许。 本文我们将来讨论一下什么是僵尸进程,僵尸进程是怎么产生的,如何杀死一个僵尸进程。 Linux中的进程是什么? 讲到进程,我们要先了解一下另一个概念: &…

    Linux 2023年6月14日
    0151
  • rsync实时同步

    环境说明 服务器类型 IP地址 应用 操作系统 源服务器 192.168.169.139 rsync inotify-tools 脚本 centos8 目标服务器 192.168….

    Linux 2023年6月13日
    0109
  • 兼容各种浏览器的上下滚动代码

    直接切入正题 红色的表示为要注意统一的。 蓝色是表示要更改的。 内容高度一定要大于box1的高度否则不会滚动,本框架用的是phpcms,大家可根据自己的框架更改循环。 | {pc:…

    Linux 2023年6月13日
    086
  • 😊🙈使用unicode字符集显示emoji表情

    无意中看到Github上很多readme.md用了漂亮又有趣的表情符号,想着是怎么实现。开始我还以为是什么emoji的插件,查着查着才知道,原来unicode字符集已经加入了emo…

    Linux 2023年6月13日
    0100
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球