拓扑排序总结

AOV网与拓扑排序

㈠AOV网概念

            在有向图中,若用顶点表示活动,有向边表示活动顺序,称该有向图为顶点表示活动的网络,简称AOV网。在AOV网中,若从顶点i到顶点j之间存在一条有向路径,称顶点i是顶点j的前驱,或顶点j是顶点i的后继。若<i,j>是图中的边,则称i是j的直接前驱,j是i直接后继。

㈡AOV网的实际意义

             在一个大的工程中,常被划分为许多较小的子工程,在整个工程实施过程中,有些活动的开始是以它的所有前序活动结束为先决条件的,而有些则没有先决条件,可以安排在任意时间开始。AOV网就是一种可以形象的反映出整个工程中各个活动之间前后关系的有向图。

㈢拓扑排序

在有向图G=(V,E)中,对于V中顶点的线性序列,若在G中从顶点Vi到Vj有一条路径,则在序列中Vi必在Vj之前,称该序列为G的一个拓扑序列。构造有向图的一个拓扑序列的过程称为称为拓扑排序。

说明:

⑴在AOV网中,若不存在回路,则所有活动可排成一个线性序列,使得每个活动的所有前驱活动都排在该活动的前面,那么该序列称为拓扑序列。

⑵拓扑序列不是唯一的。

⑶AOV网并不一定都有拓扑序列。AOV网中不能出现有向回路(或有向环),若出现了有向环,则意味着某项活动以自己为先决条件,这是不对的,工程将无法进行。程序将出现死循环。因此,对于给定的AOV网,应判断它是否存在有向环。判断方法是对AOV网进行拓扑排序,若该序列中包含AOV网中全部顶点,则无环,否则存在有向环,该网代表的工程是不可行的。

⑷拓扑排序的实际意义为:若按照该拓扑序列的顶点次序进行每项活动,就能保证进行每一项活动时,其所有前驱活动均已完成,从而使工程顺利执行。

㈣拓扑排序算法步骤

             ⑴在AOV网中选取一个入度为0的顶点(没有前驱),且输出之;

⑵在AOV网中删除此顶点及该顶点发出来的所有有向边;

⑶重复⑴、⑵两步,直至AOV网中所有顶点均输出或不存在入度为0的点。

参考代码:

代码一,二维数组建图;

#include<stdio.h>
#include<string.h>
int map[1010][1010];
int indegree[1010];
int ans[1010];
void toposort(int n)
{
	for(int i=1;i<=n;i++)					//对每个点进行查找
	{
		int j=1;
		while(indegree[j])					//从小到大查找到无前驱的
			j++;
		ans[i]=j;							//存放到数组中,也可直接输出之
		indegree[j]=-1;						//将入度记为-1,防止再次被查找到
		for(int k=1;k<=n;k++)				//更新与查找到的点有关的点的前驱
			if(map[j][k])
				indegree[k]--;
	}
}
int main()
{
	int n,m;								//有n个点,m条边
	while(scanf("%d%d",&n,&m)!=EOF)
	{
		memset(map,0,sizeof(map));			//初始化
		memset(indegree,0,sizeof(indegree));
		for(int i=0;i<m;i++)
		{
			int p1,p2;
			scanf("%d%d",&p1,&p2);
			if(!map[p1][p2])				//去重操作
			{
				indegree[p2]++;				//存放p2的前驱值
				map[p1][p2]=1;				//p2的前驱为p1,做标记
			}
		}
		toposort(n);
		for(int i=1;i<n;i++)				//ans数组中存放的是一种拓扑序列
			printf("%d ",ans[i]);
		printf("%d\n",ans[n]);
	}
	return 0;
}

代码二,链式建图;

#include<stdio.h>
#include<string.h>
int head[5100];
int indegree[5100];
int ans[5100];
struct node{
	int to;
	int next;
}a[5100];
void toposort(int n)
{
	for(int i=1;i<=n;i++)
	{
		int j=1;
		while(indegree[j])
			j++;
		ans[i]=j;
		indegree[j]=-1;
		for(int k=head[j];k!=-1;k=a[k].next)
		{
			indegree[a[k].to]--;
		}
	}
}
int main()
{
	int n,m;
	while(scanf("%d%d",&n,&m)!=EOF)
	{
		memset(head,-1,sizeof(head));
		memset(indegree,0,sizeof(indegree));
		for(int i=1;i<=m;i++)
		{
			int p1,p2;
			scanf("%d%d",&p1,&p2);
			a[i].to=p2;
			a[i].next=head[p1];
			head[p1]=i;
			indegree[p2]++;
		}
		toposort(n);
		for(int i=1;i<n;i++)
			printf("%d ",ans[i]);
		printf("%d\n",ans[n]);
	}
	return 0;
}

代码三,队列实现;

#include<stdio.h>
#include<string.h>
#include<queue>
using namespace std;
int degree[5100];
int map[510][510];
void toposort(int n)
{
	priority_queue<int,vector<int>,greater<int> >q;
	for(int i=1;i<=n;i++)
		if(degree[i]==0)
			q.push(i);
	int flag=1;
	while(!q.empty())
	{
		int top=q.top();q.pop();
		degree[top]=-1;
		if(flag)
			printf("%d",top);
		else
			printf(" %d",top);
		flag=0;
		for(int j=1;j<=n;j++)
			if(map[top][j])
			{
				degree[j]--;
				if(degree[j]==0)
				{
					q.push(j);
				}
			}
	}
	printf("\n");
}
int main()
{
	int n,m,p1,p2;
	while(scanf("%d%d",&n,&m)!=EOF)
	{
		memset(map,0,sizeof(map));
		memset(degree,0,sizeof(degree));
		for(int i=0;i<m;i++)
		{
			scanf("%d%d",&p1,&p2);
			if(map[p1][p2]==0)
			{
				map[p1][p2]=1;
				degree[p2]++;
			}
		}
		toposort(n);
	}
	return 0;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-06 11:02:56

拓扑排序总结的相关文章

拓扑排序讲解

在这里我们要说的拓扑排序是有前提的 我们在这里说的拓扑排序是基于有向无环图的!!!. (⊙o⊙)…我所说的有向无环图都知道是什么东西吧.. 如果不知道,我们下面先来来说说什么是有向无环图. 所谓有向无环图,顾名思义是不存在环的有向图(至于有向图是什么不知道的在前面我们有一个图论讲解上都有). 点的入度:以这个点为结束点的边数. 点的出度:以这个点为出发点的边的条数. 拓扑序就是对于一个节点的一个排列,使得(u,v)属于E,那么u一定出现在v的前面.然而拓扑排序就是一个用来求拓扑序的东西. 对于左

CSU 1804: 有向无环图(拓扑排序)

http://acm.csu.edu.cn/csuoj/problemset/problem?pid=1804 题意:…… 思路:对于某条路径,在遍历到某个点的时候,之前遍历过的点都可以到达它,因此在这个时候对答案的贡献就是∑(a1 + a2 + a3 + ... + ai) * bv,其中a是之前遍历到的点,v是当前遍历的点. 这样想之后就很简单了.类似于前缀和,每次遍历到一个v点,就把a[u]加给a[v],然后像平时的拓扑排序做就行了. 1 #include <bits/stdc++.h>

7-9-有向图无环拓扑排序-图-第7章-《数据结构》课本源码-严蔚敏吴伟民版

课本源码部分 第7章  图 - 有向无环图拓扑排序 ——<数据结构>-严蔚敏.吴伟民版        源码使用说明  链接??? <数据结构-C语言版>(严蔚敏,吴伟民版)课本源码+习题集解析使用说明        课本源码合辑  链接??? <数据结构>课本源码合辑        习题集全解析  链接??? <数据结构题集>习题解析合辑        本源码引入的文件  链接? Status.h.SequenceStack.c.ALGraph.c    

hihoCoder 1175:拓扑排序二

题目链接: http://hihocoder.com/problemset/problem/1175 题目难度:一星级(简单题) 今天闲来无事,决定刷一道水题.结果发现这道水题居然把我卡了将近一个钟头. 最后终于调通了.总结起来,原因只有一个:不够仔细. 思路不用细说了,就是拓扑排序的简单应用.然而,一些不起眼的细节才是让你掉坑里的真正原因. 猜猜哪儿可能出bug? // A simple problem, but you can't be too careful with it. #inclu

hdu1285(拓扑排序)

这道题要求没有输赢关系的两个元素必须按照升序输出,有输赢关系的,赢得在输的前面,所以用队列或者栈来降低时间复杂度的优化过的拓扑排序会出错. 比如这组输入 5 3 1 2 2 3 4 5 至少我写的两种拓扑排序都wa了.但是不用队列或者栈来优化的话, 1.每次都从头至尾扫描一遍,找到一个没标记过的节点, 2.将它标记 3.然后删除从它出来的每条边. 重复这三个操作,加标记的次序,就是题目要的答案. 下面的代码中用到了队列,但只是用来保存答案而已.并没有用它优化的意思. #include <iost

uva 10305 Ordering Tasks(拓扑排序)

拓扑排序,不用判断是否有环,dfs挺简单的 代码: #include<stdio.h> #include<string.h> #include<stdlib.h> int map[105][105]; int visit[105]; int c[105]; int n,m,t; void dfs(int x) { visit[x] = 1; for(int i=1; i<=n; i++) { if(!visit[i]&&map[i][x]==1)

NOJ 2015年陕西省程序设计竞赛网络预赛(正式赛)(忙碌的选课系统-拓扑排序注意重边)

D - 忙碌的选课系统 Time Limit: 10000 ms        Memory Limit: 65536 KB Submit Description 每学期末,都是万众瞩目的选课时间,由于人数过多,某学校的服务器常常被无数的学生挤的爆掉,这是,教务系统大人说,你们选个课都这么慢,居然还怪我们.于是,每次教务系统都会在服务器快要瘫痪前关闭它.在无数学生的强烈抗议下,教务系统妥协了,再给每个人一次机会,但他让我们用最快的方式决定该选的课程,选上后就退出. 这让大一学渣狗犯了难,在新的选

POJ1420 Spreadsheet(拓扑排序)注意的是超内存

Spreadsheet Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 617   Accepted: 290 Description In 1979, Dan Bricklin and Bob Frankston wrote VisiCalc, the first spreadsheet application. It became a huge success and, at that time, was the ki

拓扑排序之变量序列算法分析

拓扑排序之变量序列 巧若拙(欢迎转载,但请注明出处:http://blog.csdn.net/qiaoruozhuo) 题目描述: 假设有n个变量(1<=n<=26,变量名用单个小写字母表示),还有m个二元组(u,v),分别表示变量u小于v.那么,所有变量从小到大排列起来应该是什么样子的呢? 例如有4个变量a,b,c,d,若以知a<b,c<b,d<c,则这4个变量的排序可能是a<d<c<b.尽管还有可能其他的可能,你只需找出其中的一个即可. 输入: 输入为一

UVALive 6467 Strahler Order 拓扑排序

这题是今天下午BNU SUMMER TRAINING的C题 是队友给的解题思路,用拓扑排序然后就可以了 最后是3A 其中两次RE竟然是因为: scanf("%d",mm); ORZ 以后能用CIN还是CIN吧 QAQ 贴代码了: 1 #include <stdio.h> 2 #include <string.h> 3 #include <stdlib.h> 4 #include <math.h> 5 #include <iostre