C/C++浅析邻接表拓扑排序算法的实现

 更新时间:2022年07月27日 10:20:44   作者:菠菠萝宝  
这篇文章主要介绍了C/C++对于邻接表拓扑排序算法的实现,邻接表是图的一种链式存储方法,其数据结构包括两部分:节点和邻接点

前言

在软件开发、施工过程、教学安排等等的一系列活动中,往往需要一个有向无环图来表示其是否成成功进行下去。

在一个有向图为顶点表示活动的网中,我们称为AOV网(Activity On Vertex Network)。设G={V,E}是一个具有n个顶点的有向图,V中的顶点序列v1,v2,…,vn,满足若从顶点vi到vj有一条路径,则在顶点序列中顶点vi必在vj之前。则我们称这样的顶点为一个拓扑序列。

所谓拓扑排序,其实就是对一个有向图构造拓扑序列的过程。如果所有的顶点被输出,则说明有向图中不存在回路,反之则是有回路。

一、拓扑排序算法的思路

拓扑排序往往用在有向邻接表中,这里也就只用有向邻接表来实现。

先找出所有节点的入度。

再在AOV网中选择一个入度为0的顶点输出,然后删除此顶点,将其连接的节点的入度减一直至输出所有顶点或者AOV网中不存在入度为0的顶点为止。

二、实现步骤

1.求个顶点的入度

设置一个indegree数组来存放各个顶点的入度。

int* indegree = (int*)malloc(sizeof(int) * G.vexnum);
//对单个节点p求入度
void CountIndegree(AdjList g, int* indegree, ArcNode* p) {
	while (p != NULL) {
		indegree[p->adjvex]++;
		p = p->nextarc;
	}
	return;
}

2.拓扑排序的实现

这里对栈的使用还是调用stl中的stack,比较方便。

bool TopoSort(AdjList g, int* indegree) {
	//先清空申请的indegree数组,或者也可以在初始化时采用calloc,就不用在这里置为0了
	for (int i = 0; i < g.vexnum; i++) {
		indegree[i] = 0;
	}
	//遍历边表中的每一个顶点,用CountIndegree()遍历单个节点
	for (int i = 0; i < g.vexnum; i++) {
		ArcNode* p = g.vertexlist[i].firstarc;
		CountIndegree(g, indegree, p);
	}
	stack<int>S;
	//如果该顶点的入度为0,则入栈。
	for (int i = 0; i < g.vexnum; i++) {
		if (indegree[i] == 0) {
			S.push(i);
		}
	}
	//count用来表示已经输出的节点个数
	//如果所有的顶点被输出,则count==g.vexnum,无回路,反之count<g.vexnum,则是有回路。
	int count = 0;
	while (!S.empty()) {
		int top = S.top();
		printf("%c ", g.vertexlist[top].data);
		S.pop();
		count++;
		ArcNode* p = g.vertexlist[top].firstarc;
		for (p; p != NULL; p = p->nextarc) {
			int i = p->adjvex;
			if (--indegree[i] == 0) {
				S.push(i);
			}
		}
	}
	if (count == g.vexnum) {
		return true;
	}
	return false;
}

三、测试结果

自己花了一个看起来挺复杂的图,一下也看不出来有没有环

首先算一算入度,顺带打印一下。

接下来是拓扑排序的结果

完美!

总结

每个顶点进栈一次出战一次,度减一的操作执行了e次,所以整个算法的时间复杂度为O(n+e)。

到此这篇关于C/C++浅析邻接表拓扑排序算法的实现的文章就介绍到这了,更多相关C++拓扑排序内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言字符串旋转问题的深入讲解

    C语言字符串旋转问题的深入讲解

    这篇文章主要给大家介绍了关于C语言字符串旋转问题的相关资料,文中给出了详细的实现方法,并对每种方法进行了分析和示例代码,需要的朋友可以参考下
    2021-09-09
  • C语言实现最小生成树构造算法

    C语言实现最小生成树构造算法

    这篇文章主要为大家详细介绍了C语言实现最小生成树构造算法,利用Prim算法或kruskal算法求解,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • C语言中常见的六种动态内存错误总结

    C语言中常见的六种动态内存错误总结

    学习过C语言中的动态内存函数,例如【malloc】、【calloc】、【realloc】、【free】,那它们在使用的过程中会碰到哪些问题呢,本本文我们一起来探讨下,感兴趣的朋友跟着小编一起来看看吧
    2023-11-11
  • C语言实现大顶堆的示例代码

    C语言实现大顶堆的示例代码

    最大堆,又称大根堆(大顶堆)是指根结点(亦称为堆顶)的关键字是堆里所有结点关键字中最大者,属于二叉堆的两种形式之一。本文将用C语言实现大顶堆,感兴趣的可以了解一下
    2022-07-07
  • C++中使用function和bind绑定类成员函数的方法详解

    C++中使用function和bind绑定类成员函数的方法详解

    这篇文章主要介绍了C++中使用function和bind绑定类成员函数的方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-11-11
  • C语言实现静态版通讯录的示例代码

    C语言实现静态版通讯录的示例代码

    这篇文章主要为大家详细介绍了如何利用C语言实现一个简单的静态版通讯录,文中的示例代码讲解详细,对我们学习C语言有一定帮助,需要的可以参考一下
    2022-08-08
  • 关于C++智能指针shared_ptr和unique_ptr能否互转问题

    关于C++智能指针shared_ptr和unique_ptr能否互转问题

    C++中的智能指针最常用的是shared_ptr和unique_ptr,C++新手最常问的问题是我从一个函数中拿到unique_ptr,但要转成shared_ptr才能使用,要怎么转换?同理是否能将shared_ptr转换成unique_ptr,面对这些问题,跟随小编一起看看吧
    2022-05-05
  • C++中 map的基本操作

    C++中 map的基本操作

    map是一类关联式容器。接下来通过本文给大家分享c++中的map基本操作,需要的朋友参考下
    2017-05-05
  • C语言之qsort函数详解

    C语言之qsort函数详解

    这篇文章主要介绍了C语言中qsort函数的用法实例详解的相关资料,希望通过本文能帮助到大家,让大家理解掌握这部分内容,需要的朋友可以参考下
    2021-08-08
  • C++实现的打字母游戏示例

    C++实现的打字母游戏示例

    这篇文章主要介绍了C++实现的打字母游戏,涉及C++字体操作、时间及键盘响应相关操作技巧,需要的朋友可以参考下
    2017-08-08

最新评论