拓扑排序(Topological svorting)

级别:提高级
难度:6

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

拓扑排序是存在在DAG中,而在DAG中因为是无环的,所以必然存在一个点v1他是入度为0的,那么这个入度为0的点就是当前的出发点。注意:拓扑排序有时结果不唯一。

1.找一个入度为零的端点,如果有多个,则从编号小的开始找;
2.将该端点的编号输出;
3.将该端点删除,同时将所有由该点出发的有向边删除;
4.循环进行 1,2 和 3 ,直到图中的图中所有点的入度都为零;
5.拓扑排序结束;

Introduction to Topological Sort - LeetCode Discuss


代码实现:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
/**************************************************************** 
 * Description: topoSort
 * Author: Alex Li
 * Date: 2023-06-30 21:49:30
 * LastEditTime: 2024-01-29 23:15:02
****************************************************************/
#include <iostream>
#include <vector>
#include <queue>
using namespace std;

int n,m;
const int N=1000;
vector<int>  e[N],tp; //e[x]存点x的邻点,tp存拓扑序列;
int din[N];   //存结点的入度

bool TopoSort(){
     queue<int>  q;
     for (int i =1; i <=n; i++){
        if(din[i]==0) q.push(i);//将入度为0的结点入到q队列里。
        
     }
     while(q.size()){
        int x=q.front();  //把q队列首元素赋值给x
        q.pop();   //首元素出q队列
        tp.push_back(x);  //入到tp序列
    for (int i = 0; i < e[x].size(); i++){
        ////将刚出列x结点所有相联的结点的入度减1,如果入度是0,就入到q队列里
        if(--din[e[x][i]]==0)q.push(e[x][i]); 
          // for (auto y:e[x]) if(--din[y]==0)q.push(y);
        }
    }
     return tp.size()==n;  //如果拓扑序列个数和元素个数一样,说明可以拓扑排序 
     
}
int main(){
    int a,b;  //n是指结点数,m是边数
    cin>>n>>m;
    for (int  i = 0; i < m; i++){
        cin>>a>>b;
        e[a].push_back(b);  
        din[b]++;  //记录没个节点的入度
    }
    if(!TopoSort()) puts("TopoSort is not supported");
    else 
        for (int  i = 0; i <n; i++)cout<<tp[i];
return 0;        
}

测试数据:

6 7
1 2
1 4
2 3
4 2
4 6
4 5
5 6
输出:142536

练习:P3644

Scroll to Top