拓扑排序

基本概念

定义:由某个集合上的一个偏序得到该集合上的全序的过程。

离散数学相关概念

  • 偏序:若集合X上的关系R是自反的、反对称的和传递的、则称R是集合X上的偏序关系。

  • 全序:设R是集合X上的偏序。如果对每个x,y∈X必有xRy或yRx,则称R是集合X上的全序关系。

    偏序与全序的有向图

    由上图所示在(b)图表示全序,而(a)的有向图上人为地加上一个表示V2<V3的弧(表示V2领先于V3)则(a)表示的亦为全序,这个全序称为拓补有序,由偏序定义得到拓补有序的操作便是拓补排序。

  • 表示偏序的有向图可以用来表示流程图。图中的每一条边表示两个子工程之间的次序关系。

AOV网(Activity on vertex Network)

定义:用顶点表示活动,用弧表示活动间的优先关系的有向图称为定点表示活动的网。顾名思义。

  • 在网中,若从顶点i到另一顶点j之间有一条有向路径,则称i为j的前驱,j为i的后继。

  • 在AOV网中不应该出现有向环,因为存在环意味着某项活动会以自己为先决条件,这是不对的。因此要对给定的AOV网进行判断是否有环。检测方法就是对AOV网进行拓补排序,若所有顶点都在拓补有序序列中,则AOV网中必没有环。

拓补排序的实现

  1. 在有向图中选择没有前驱的点的顶点并将其输出。
  2. 从图中删除该顶点和所有以它为尾的弧。
    重复上面的两部,直到所有顶点均已经输出。
  • 采用邻接表作有向图的存储结构。
  • 声明存放顶点入度的数组 Indegree,入度为零的顶点即为没有前驱的顶点。
  • 删除顶点以及以它为尾的弧的操作,则可更换为弧头顶点入度减一的操作。
  • 为了避免重复检测入度为零的情况。另设一个线性表(栈或队列,根据题内特殊要求而定,就拓补排序定义而言无差别)暂存所有在删除弧操作后产生的新的入度为零的点。

复杂度分析

对于有n个顶点和e条弧的有向图而言总的时间复杂度为O(n+e)

代码

#include <stdio.h>
#include <stdlib.h>
#define MAX_VERTEX_NUM 40
//图的邻接矩阵表示方式
typedef enum {DG,DN,UDG,UDN} GraphKind;
typedef struct ArcNode{
    int adjvexNum; //该弧所指向的顶点在图的定点结构体组中的下角标
    struct ArcNode *nextarc;
}ArcNode;
typedef struct VexNode{
    ArcNode *firstarc;
    char data;
}VexNode,AdjList[MAX_VERTEX_NUM];
typedef struct {
    AdjList vertice;
    int vexnum,arcnum;  // 图当前顶点数和边数
    GraphKind kind; //图的种类类型
}MGraph;
int Visited[MAX_VERTEX_NUM] = {0};
int Indegree[MAX_VERTEX_NUM] = {0};
void Insert(char a,char b,MGraph &G){
    int vex1,vex2;
    for(int i = 0; i < G.vexnum; i++){
        if(a == G.vertice[i].data){
            vex1 = i;
        }
        if(b == G.vertice[i].data){
            vex2 = i;
        }
    }
    Indegree[vex2]++;
        ArcNode *arcNode = (ArcNode*)malloc(sizeof(ArcNode)); //初始化所有边 
        arcNode->nextarc = NULL;
        arcNode->adjvexNum = vex2;
        
        if(G.vertice[vex1].firstarc == NULL){    //将边加入到邻接表 
            G.vertice[vex1].firstarc = arcNode;
        }
        else{
            ArcNode *t = NULL;
            ArcNode *pre = NULL;
            for(t = G.vertice[vex1].firstarc;t != NULL;t = t->nextarc){
                pre = t;
            }
            pre->nextarc = arcNode;
        }
}

void CreateGraph(MGraph &G){
    //scanf("%d %d",&G.vexnum,&G.arcnum);
    G.vexnum = 11;
    G.arcnum = 21;
    for(int i = 0; i < G.vexnum; i++) {  //初始化所有顶点
        G.vertice[i].firstarc = NULL;
    }
    G.vertice[0].data = 'S';
    G.vertice[1].data = 'A';
    G.vertice[2].data = 'B';
    G.vertice[3].data = 'C';
    G.vertice[4].data = 'D';
    G.vertice[5].data = 'E';
    G.vertice[6].data = 'F';
    G.vertice[7].data = 'G';
    G.vertice[8].data = 'H';
    G.vertice[9].data = 'I';
    G.vertice[10].data = 'T';
    //for(int k = 0; k < G.arcnum; k++){
    // int vex1,vex2,weight;  //输入一条边的起点和终点以及这条边的权值 
    //  scanf("%d %d",&vex1,&vex2); 
    //}
    char a[21]={
        'S','S','S','A','A','B','C','D','D','E','E','E','F','F','G','G','G','H','H','I','I'
    };
    char b[21]={
    'A','D','G','B','E','C','T','A','E','C','F','I','C','T','D','E','H','E','I','F','T'
    };
    for(int i = 0; i < G.arcnum; i++){
            Insert(a[i],b[i],G);
    }
}
typedef struct Node{
    int data;
}QNodes[100];
typedef struct Queue{
    int head;
    int tail;
    QNodes qNodes;
}Queue;
void TopologicalSort(MGraph G){
    int TopNum[MAX_VERTEX_NUM] = {0};
    Queue queue;
    queue.tail = 0;
    queue.head = 0;
    int counter = 0;
    for(int i = 0; i < G.vexnum;i++) {
        if(Indegree[i] == 0) {
            queue.qNodes[queue.tail].data = i;
            queue.tail++;
        }
    }
        while(queue.tail != queue.head){
            int p = queue.qNodes[queue.head].data;
            queue.head++;
            TopNum[counter] = p;
            counter++;
            for(ArcNode *t = G.vertice[p].firstarc;t;t = t->nextarc){
                Indegree[t->adjvexNum]--;
                if(!Indegree[t->adjvexNum]){
                    queue.qNodes[queue.tail].data = t->adjvexNum;
                    queue.tail++;
                }
            }
        }
    for(int i = 0; i < counter; i++){
          printf("%c,",G.vertice[TopNum[i]].data);
    } 
}
int main(){
    MGraph graph;
    CreateGraph(graph);
    TopologicalSort(graph);
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351