一个有向无环图的拓扑排序序列是唯一的么
一般不唯一,如果同时有多个如果为0的顶点供选择时,不会唯一
设图 G 采用邻接表存储,则拓扑排序算法的时间复杂度为()
如果是邻接表存储,拓扑排序算法的时间复杂度应该是O(n + e),n是顶点个数,e是弧的数量
关键路径为什么要用拓扑排序
算法效率、时间复杂度问题,你的方法也可以,但是学算法的目的是比较各种算法的效率,拓扑明显是好的
采用邻接表存储,拓扑排序算法的时间复杂度为多少?
要看使用什么样的拓扑排序,最好的方法是输出DFS的逆序,这样的算法复杂度是O(V+L),V是顶点个数,L是边个数。
C++假设给我们一个任意的图,它可能是也可能不是DAG(有向无圈图),推广拓扑排序算法
我可以给你一个思路,先dfs判断这个图有没有环,如果是DAG就直接拓扑排序,如果不是DAG的话可以考虑将有向图删去一些边,变成一棵树,考虑边A指向B,可以理解为A是B的父亲节点,然后dfs一下就能得到一棵树,接下来逐一枚举每条边C指向D,如果树上C与D间没有边,就输出树上从C到D路径上所有边和C指向D的这条边
一个有向无环图的拓扑排序序列是唯一的么
一般都不唯一,如果某步有超过1个入度为0的顶点供选择,这时序列肯定不唯一
在拓扑排序算法中用堆栈和用队列产生的结果会不同吗
抽取关键信息的结构是有所不同的,毕竟一个是栈存储中间信息,一个是队列存储中间信息。面对不同的拓扑排序结构,有的结构两种实现方法的拓扑排序功效相同,有的则会不同
拓扑排序的非计算机应用
拓扑排序常用来确定一个依赖关系集中,事物发生的顺序。例如,在日常工作中,可能会将项目拆分成A、B、C、D四个子部分来完成,但A依赖于B和D,C依赖于D。为了计算这个项目进行的顺序,可对这个关系集进行拓扑排序,得出一个线性的序列,则排在前面的任务就是需要先完成的任务。注意:这里得到的排序并不是唯一的!就好像你早上穿衣服可以先穿上衣也可以先穿裤子,只要里面的衣服在外面的衣服之前穿就行。
拓扑排序:已知有九门课程,依次编号为C0至C8,在图一中给出了给出了各门课程之间先后关系。例如:C0是C2
我最近喜欢上为那些真诚提问问题的人们供献一份自己的绵薄之力。在您们问,我回答的过程中,相信我们不仅仅是问问答答,更美妙的是我们在一问N答的过程中充分尝到了分享问题分享答案的喜悦之情,我们彼此都得到了很多。不过,在为您们正要解答与解答完后的时间里,总是多多少少会出些状况:我在这里先承诺,我会在以后为您们答题的过程中,竭尽全力做到,不马虎,能多详细就多详细得答。所以请您们在提问问题时,要让我们答题人能够一目了然,知道您们写的是什么,然后在把题目完整的前提下,让我们看到的题目和您们想让我们解答的完全一样(主要叙述与数据千万不能错)。最后,让我们为自己能在人与自然和谐相处的世界里更加"给力"共同给力吧。^_^转载于"真诚的给提问者的一些话100分"
如何利用拓扑排序将一个有向无环图的邻接矩阵中的非零元素集中到对角线以上
二者的区别:邻接矩阵(Adjacency Matrix):是表示顶点之间相邻关系的矩阵。设G=(V,E)是一个图,其中V={v1,v2,…,vn}。G的邻接矩阵是一个具有下列性质的n阶方阵:①对无向图而言,邻接矩阵一定是对称的,而且主对角线一定为零(在此仅讨论无向简单图),副对角线不一定为0,有向图则不一定如此。②在无向图中,任一顶点i的度为第i列所有元素的和,在有向图中顶点i的出度为第i行所有元素的和,而入度为第i列所有元素的和。③用邻接矩阵法表示图共需要n^2个空间,由于无向图的邻接矩阵一定具有对称关系,所以扣除对角线为零外,仅需要存储上三角形或下三角形的数据即可,因此仅需要n(n-1)/2个空间。
数据结构题。有向图,给出该图的一种拓扑排序序列
拓扑排序的方法和步骤:(1)在图中选一个没有前趋的顶点并输出之(2)删除该顶点及由它发出的各边,直到图中不存在没有前趋的顶点为止。答案:1,3,2,4,5
数据库大神来啊、给出下列AOV网的可能的拓扑排序序列。
针对该题,可能的拓扑排序:1.C->D->B->A->E2.D->B->C->A->E3.D->C->B->A->E当然,拓扑序列不一定唯一如果图中,这里是AOV网中存在有向环,则无法完成拓扑排序。
在拓扑排序的过程中,每个记录的i/o次数必定相等对吗
boolNetwork::Topological(intv[]){//计算有向图中顶点的拓扑次序//如果找到了一个拓扑次序,则返回true,此时,在v[0:n-1]中记录拓扑次序//如果不存在拓扑次序,则返回falseintn=Vertices();//计算入度int*InDegree=newint[n+1];InitializePos();//图遍历器数组for(inti=1;i<=n;i++)//初始化InDegree[i]=0;for(i=1;i<=n;i++){//从i出发的边intu=Begin(i);while(u){InDegree[u]++u=NextVertex(i);}}//把入度为0的顶点压入堆栈LinkedStackS;for(i=1;i<=n;i++)if(!InDegree[i])S.Add(i);//产生拓扑次序i=0;//数组v的游标while(!S.IsEmpty()){//从堆栈中选择intw;//下一个顶点S.Delete(w);v[i++]=w;intu=Begin(w);while(u){//修改入度InDegree[u]--;if(!InDegree[u])S.Add(u);u=NextVertex(w);}}DeactivatePos();delete[]InDegree;return(i==n);}
拓扑排序序列倒着写是逆拓扑排序序列吗
拓扑排序的方法是,先找到第一个没有被指的,就是C1,加入序列。然后擦掉跟C1有关的边,此时C2和C3都满足没有被指,选一个,比如选C2,加入序列,擦掉和C2有关的边,这个时候可以选C3,C4,C5或C6……如此而已
拓扑排序是一种内部排序的算法,对吗?
不是,不对的
说明拓扑排序与冒泡排序概念的区别
冒泡排序是一种交换排序方式。设有n个数据依次放在数组元素a(1)至a(n)中,用冒泡法对这n个数据进行递增排序的过程为:先比较a(1)与a(2),若逆序则交换之,接着比较a(2)与a(3),若逆序就交换……依次进行,知道将a(n-1)与a(n)比较交换完,才算完成...
有向无环图的判定及拓扑排序
五分您打发谁呢?数据结构要好好学啊
关于C语言拓扑排序的问题,哪位大侠帮帮忙啊,谢谢!
时间有点紧,就不能给你写代码了,不好意思,就说说思路吧。 存储结构就用最简单的数组吧,数组中有三个元素,编号,名称,先修课程。 我们假设把一系列的又关联性的课程炼成一串, 如;c1->c2->c3.这样的话,上述问题其实就转化成来这样的一个问题: 串起来之后哪个链表最长,以及怎么样安排课程才能让所有链表最短。因为假设每门课的时间时一样的。这样的话,我们打印出来信息一目了然,一个while循环,当他的先修不为空,输出他本身,他等于他的先修,循环下去。我们可以看见每一门课的链表。解决来第一个问题。更加知道了最基础的课程时那些。也就时没有先修课程的。 然后怎么样安排课程链表最短,这个需要知道每个学期最多安排的课程数。然后就时先修最基本的课程,然后逆着上面的链表开课。
n 个顶点的有向图用邻接矩阵 array 表示,下面是其拓扑排序算法,试补充完整。
(1) 0 (2) array[j][i](3)0 (4)indegree[i] == 0(5)vex(6)array[k][i] == 1(7)indegree[i] == 0(8) count < n
图的拓扑排序是不是不唯一的?
同一层次(同时会有多个点入度为0)时拓扑排序不唯一
如何用C++语言实现AOV网的所有拓扑排序?(最好能实现动态演示)
#define max 图的顶点数//弧节点的结构typedef struct EdgeNode{int adjvex;struct EdgeNode *next;};//顶点节点的结构typedef struct Vnode{VertexType vertex;//顶点数据类型EdgeNode *link;}Adjlist[max];//拓扑排序算法描述void topsort(Adjlist g)//假设G有n个顶点、e条边的有向图,g是他的邻接表,每个节点设两个//域vex,next,对入度为0的顶点设计带链的栈,top指示栈的指针,in为入度{readlist();//输入e条弧并建立邻接表top=0;for(i=1;i<=n;i++)//查入度为0的顶点,并建立链栈if(g[i].in==0){g[i].in=top;top=i;}m=0;//设m为计数器计算输出的顶点个数while(top!=0){j=top;top=g[top].in;//退栈printf("%d",j);m++;//输出顶点并计数q=g[j].link;//q是指针,指示以j为尾的弧while(q!=NULL){k=q->vex;//顶点k为j的直接后继g[k].in=g[k].in-1;//入度减1if(g[k].in==0){g[k].in=top;top=k;//入度为0的顶点进栈}q=p->next;}}if(m<n)printf("the network have cycle");//输出顶点数不足n,说明网中无环}
在用邻接表表示图时,拓扑排序算法时间复杂度为多少?
设图中顶点n个,弧e条,则在邻接表上进行拓扑排序的时间复杂度为O(n + e)
拓扑排序结果序列中的第一个节点一定是入度大于0的点?
那个选择题的答案错了,根据拓扑排序的原则,序列的第一个节点一定是没有前驱的,入度等于0的,不然怎么能够拓扑排序排在第一位呢,排在第一位也就是说没有任何前导的结点,入度一定为0了
怎样通过拓扑排序判断图是否有环
拓扑排序的核心就是每次找入度为0的点 进入输出队列 然后将与此点相连的节点入度减1 重复做当做n-1 次后还有点没进输出队列 那么这些点就是环上的 因为环上的各点入度都为1 没有0的 就不能更新
求解析:图G=(V,E)写出图G中顶点的所有拓扑排序。
就是按答案的顺序,E中的逗号前面的都在逗号后的数字的前面。
堆排序的代码(拓扑排序的伪代码)
额。。程序最后多打了一个大括号这段代码的功能是对一个有向图进行拓扑排序,若有环则抛出异常首先要明白拓扑排序是做什么的,详见百度百科--拓扑排序(有图有样例)然后说这段代码:通过将图中入度为0的点存入队列,用队首的点去查找与此点连接且入度为1的点,把找到的点在存入队列,令这队首的点出队。如此至到找不到入度为0的点,结束while循环,最后再判断进入再离开队列的点的个数若小于图中总点数,则判定有环。注:可以根据注释判断函数功能void Graph::topsort(){ Queue<Vertex> q;//定义存储点的信息的队列,由于是伪代码,信息类型不详,一般是该点在题目中的编号 int counter=0;//记录当前有几个点已经出队了 q.makeEmpty();//初始化队列,令其清空 for each Vertex v //初始化查找入度为0的点,若找到则入队 if(v.indegree==0) //判断入度为0 q.enqueue(v); //入队 while(!q.isEmpty()) //若队列不为空则一直循环 { Vertex v=q.dequeue(); //队首点出队 v.topNum=++counter; //累加出队点数 for each Vertex w adjacent to v //用此出队点,即刚才的队首,查找仅与其相连的点,并将找到的点入队 if(--w.indegree==0) q.enqueue(w); } if(counter!=NUM_VERTICES) //若总出队点不等于图中总点数,则有环 throw CycleFoudException();}
怎样求一个有向无环图的拓扑排序序列总数
(1)设对有向无环图G=,求得它的一个拓扑序列为S,初始化S为空,然后每次从G中选取一个入度为0的点v,将v插入到S的尾部,再在G中删除点v,并删除所有以v为弧尾的边(即由v引出去的边),如此循环,直到图G中的V为空集时结束.21 2 3 4 5 6 7 81 3 2 4 5 7 6 83 1 4 2 5 6 7 83 1 2 5 4 7 8 6
【讨论】“拓扑排序算法仅适用于有向无环图”,对吗
我觉得是错的,拓扑排序也经常用来判定一个有向图是否有环,所以做为判定方法的话,肯定无论有环无环都能用的
数据结构课程设计:拓扑排序和关键路径
我有清晰的解题思路!~
拓扑排序的结果不是唯一的,请给出如图7-32所示的有向图的所有可能的拓扑序列。
由AOV网构造拓扑序列的拓扑排序算法主要是循环执行以下两步,直到不存在入度为0的顶点为止:选择一个入度为0的顶点并输出之,从网中删除此顶点及所有出边。循环结束后,若输出的顶点数小于网中的顶点数,则输出“有回路”信息,否则输出的顶点序列就是一种拓扑序列。由AOV网构造出拓扑序列的实际意义是:如果按照拓扑序列中的顶点次序,在开始每一项活动时,能够保证它的所有前驱活动都已完成,从而使整个工程顺序进行,不会出现冲突的情况。扩展资料:数据结构拓扑排序实际上是离散数学中的概念。这里不打算说太多形式化的定义,形式化的定义教科书上或者上面给的链接中就说的很详细。还是以上面选课的例子来描述这两个概念。假设在学习完了算法这门课后,可以选修机器学习或者计算机图形学。这个或者表示,学习机器学习和计算机图形学这两门课之间没有特定的先后顺序。所有可以选择的课程中,任意两门课程之间的关系要么是确定的(即拥有先后关系),要么是不确定的(即没有先后关系),绝对不存在互相矛盾的关系(即环路)。以上就是偏序的意义,抽象而言,有向图中两个顶点之间不存在环路,至于连通与否,是无所谓的。所以,有向无环图必然是满足偏序关系的。参考资料:百度百科-拓扑序列参考资料:百度百科-拓扑排序
求一个可以输出所有拓扑排序的代码或者思路,记住是所有可能的拓扑排序????????????????
伪代码:// 参数g为图,nodes为已经排序的顶点TopologicalSort(Graph g, List nodes) { g中所有没有前驱的顶点放入队列q; while(q不为空) { q中顶点出队,设为v; 复制nodes生成nodes的副本,并把v加入nodes的副本; 复制g生成g的副本,g的副本删去v和v发出的有向边; if (g的副本中没有顶点) { 输出nodes; // node为最终的排序 } else { TopologicalSort(g的副本, nodes的副本);//递归对剩下的图排序 } }}
如果具有n个顶点的有向图能够进行拓扑排序,那么有向
就此题来说,答案应该是一般。另外给出另一方面的分析,希望能对此题有所帮助。题目:若一个有向图具有有序的拓扑排序序列,那么它的邻接矩阵必定为?(比原题加了个有序的)答案是三角。(这个三角不是特殊矩阵压缩存储时的三角矩阵,而是线性代数中的三角矩阵)可以证明,对于有向图中顶点适当地编号,使其邻接矩阵为三角矩阵且主对角元全为零的充分必要条件是该有向图可以进行拓扑排序。
只有无环有向图才能进行拓扑排序吗?
是的。有环的图是相互依赖的,所以不能。
“拓扑排序算法仅适用于有向无环图”,对吗
支持7楼的说法。在书上看到的是一个拓扑排序算法,也许还有其他的方法可以进行拓扑排序。而对一个东西进行拓扑排序是要有结果的。拓扑排序算法只是一个可以进行拓扑排序的方法之一,就像各种排序算法都能排序。
拓扑排序是指结点的值是有序排列。对不对,为什么。。。。。
不是结点的值有序,是结点的逻辑先后关系保持有序
C/C++拓扑排序士兵排列问题
这个得用拓扑排序和递归来做#include <stdio.h>#include <string.h>typedef struct Sstr //方便指定字母与下标对应{ char ch; int i;}*st;char *out; //存储输出字符bool fun_s(int **a,int *in_,st p,int n,int q,int len){ int j=q; int *into=new int[n]; for (int i=0;i<n;i++) { into[i]=in_[i]; //如果不进行拷贝后面递归会把前面的值覆盖掉 } while(into[j]!=0) { j++; if (j==n) { delete []into; return true; } } if(j<n-1) fun_s(a,into,p,n,j+1,len); out[len--]=p[j].ch; //因为先读的最后一个,需要倒着存 into[j]=-1; //printf("%c",p[j].ch); if (len==-1) { for (int i=0;i<n;i++) { printf("%c",out[i]); } printf(" "); } for (int k = 0; k < n; k++) { if (a[k][j] > 0) into[k]--; } fun_s(a,into,p,n,0,len); delete []into; return true;}bool Fun_Sort(int **a,st p,const int n=1) { int i, j; int *into=new int[n]; for (i = 0; i < n; i++) { into[i]=0; for (j = 0; j < n; j++) { if (a[i][j] > 0) { into[i]++; } } } fun_s(a,into,p,n,0,n-1); delete []into; return true;}int not_exists(st a,char ch,int n){ for (int i=0;i<n;i++) { if (a[i].ch==ch) { return a[i].i; } } return -1;}int main(void){ int n,i,m,p,q; char a[1024]; int **arr; st p_str; int len=0; scanf("%d",&n); out=new char[n]; arr=new int*[n]; for (i=0;i<n;i++) { arr[i]=new int[n]; } p_str=new Sstr[n]; for(i=0;i<n;i++) { for (m=0;m<n;m++) { arr[i][m]=0; } } for (i=0;i<n;i++) { p_str[i].ch=0; p_str[i].i=i; } scanf("%s",a); m=strlen(a); for (i=0;i<m;i++) { if (a[i]>="A"&&a[i]<="Z") { p=not_exists(p_str,a[i],n); //添加字符对应 if (p==-1) { p_str[len++].ch=a[i]; } } } for (i=0;i<m;i++) { if(a[i]==">") { p=not_exists(p_str,a[i-1],n); q=not_exists(p_str,a[i+1],n); arr[p][q]=1; } else if (a[i]=="<") { p=not_exists(p_str,a[i+1],n); q=not_exists(p_str,a[i-1],n); arr[p][q]=1; } else { continue; } } Fun_Sort(arr,p_str,n); delete []p_str; for (i=0;i<n;i++) { delete []arr[i]; } delete []arr; delete []out; return 0;}
有环有向图究竟可以拓扑排序吗???
有环有向图 在工程上是不可能实际存在的,因为不可能自己即是自己的先导又是自己的后驱比如,某学校规定,修完离散数学才能修数据结构,修完数据结构以后的下一门课程一定不是离散数学了。这里的修完是指学完并考试通过,所以是重修的话还是进行不了下一道工序。修完数据结构才能修离散,但修完理论就得修数据结构,这显然是一个死循环所以有环有向图进行拓扑排序是一个伪概念,严蔚敏书上有讲的假设你有这样一个环,那么理论是可能让你一直在这里环里面做死循环的。[] 查看原帖>>
数据结构(C语言版) 图的遍历和拓扑排序
#include<string.h>#include<ctype.h>#include<malloc.h> /* malloc()等*/#include<limits.h> /* INT_MAX 等*/#include<stdio.h> /* EOF(=^Z 或F6),NULL */#include<stdlib.h> /* atoi() */#include<io.h> /* eof() */#include<math.h> /* floor(),ceil(),abs() */#include<process.h> /* exit() *//* 函数结果状态代码*/#define TRUE 1#define FALSE 0#define OK 1#define ERROR 0#define INFEASIBLE -1/* #define OVERFLOW -2 因为在math.h 中已定义OVERFLOW 的值为3,故去掉此行*/typedef int Status; /* Status 是函数的类型,其值是函数结果状态代码,如OK 等*/typedef int Boolean; Boolean 是布尔类型,其值是TRUE 或FALSE *//* .........................*/#define MAX_VERTEX_NUM 20typedef enum{DG,DN,AG,AN}GraphKind; /* {有向图,有向网,无向图,无向网} */typedef struct ArcNode{int adjvex; /* 该弧所指向的顶点的位置*/struct ArcNode *nextarc; /* 指向下一条弧的指针*/InfoType *info; /* 网的权值指针) */}ArcNode; /* 表结点*/typedef struct{VertexType data; /* 顶点信息*/ArcNode *firstarc; /* 第一个表结点的地址,指向第一条依附该顶点的弧的指针*/}VNode,AdjList[MAX_VERTEX_NUM]; /* 头结点*/typedef struct{AdjList vertices;int vexnum,arcnum; /* 图的当前顶点数和弧数*/int kind; /* 图的种类标志*/}ALGraph;/* .........................*//* .........................*//*ALGraphAlgo.cpp 图的邻接表存储(存储结构由ALGraphDef.h 定义)的基本操作*/int LocateVex(ALGraph G,VertexType u){ /* 初始条件: 图G 存在,u 和G 中顶点有相同特征*//* 操作结果: 若G 中存在顶点u,则返回该顶点在图中位置;否则返回-1 */int i;for(i=0;i<G.vexnum;++i)if(strcmp(u,G.vertices[i].data)==0)return i;return -1;}Status CreateGraph(ALGraph &G){ /* 采用邻接表存储结构,构造没有相关信息的图G(用一个函数构造4 种图) */int i,j,k;int w; /* 权值*/VertexType va,vb;ArcNode *p;printf("请输入图的类型(有向图:0,有向网:1,无向图:2,无向网:3): ");scanf("%d",&(G.kind));printf("请输入图的顶点数,边数: ");scanf("%d,%d",&(G.vexnum),&(G.arcnum));printf("请输入%d 个顶点的值(<%d 个字符): ",G.vexnum,MAX_NAME);for(i=0;i<G.vexnum;++i) /* 构造顶点向量*/{scanf("%s",G.vertices[i].data);G.vertices[i].firstarc=NULL;}if(G.kind==1||G.kind==3) /* 网*/printf("请顺序输入每条弧(边)的权值、弧尾和弧头(以空格作为间隔): ");else /* 图*/printf("请顺序输入每条弧(边)的弧尾和弧头(以空格作为间隔): ");for(k=0;k<G.arcnum;++k) /* 构造表结点链表*/{if(G.kind==1||G.kind==3) /* 网*/scanf("%d%s%s",&w,va,vb);else /* 图*/scanf("%s%s",va,vb);i=LocateVex(G,va); /* 弧尾*/j=LocateVex(G,vb); /* 弧头*/p=(ArcNode*)malloc(sizeof(ArcNode));p->adjvex=j;if(G.kind==1||G.kind==3) /* 网*/{p->info=(int *)malloc(sizeof(int));*(p->info)=w;}elsep->info=NULL; /* 图*/p->nextarc=G.vertices[i].firstarc; /* 插在表头*/G.vertices[i].firstarc=p;if(G.kind>=2) /* 无向图或网,产生第二个表结点*/{p=(ArcNode*)malloc(sizeof(ArcNode));p->adjvex=i;if(G.kind==3) /* 无向网*/{p->info=(int*)malloc(sizeof(int));*(p->info)=w;}elsep->info=NULL; /* 无向图*/p->nextarc=G.vertices[j].firstarc; /* 插在表头*/G.vertices[j].firstarc=p;}}return OK;}void DestroyGraph(ALGraph &G){ /* 初始条件: 图G 存在。操作结果: 销毁图G */int i;ArcNode *p,*q;G.vexnum=0;G.arcnum=0;for(i=0;i<G.vexnum;++i){p=G.vertices[i].firstarc;while(p){q=p->nextarc;if(G.kind%2) /* 网*/free(p->info);free(p);p=q;}}}VertexType* GetVex(ALGraph G,int v){ /* 初始条件: 图G 存在,v 是G 中某个顶点的序号。操作结果: 返回v 的值*/if(v>=G.vexnum||v<0)exit(ERROR);return &G.vertices[v].data;}int FirstAdjVex(ALGraph G,VertexType v){ /* 初始条件: 图G 存在,v 是G 中某个顶点*//* 操作结果: 返回v 的第一个邻接顶点的序号。若顶点在G 中没有邻接顶点,则返回-1 */ArcNode *p;int v1;v1=LocateVex(G,v); /* v1 为顶点v 在图G 中的序号*/p=G.vertices[v1].firstarc;if(p)return p->adjvex;elsereturn -1;}int NextAdjVex(ALGraph G,VertexType v,VertexType w){ /* 初始条件: 图G 存在,v 是G 中某个顶点,w 是v 的邻接顶点*//* 操作结果: 返回v 的(相对于w 的)下一个邻接顶点的序号。*//* 若w 是v 的最后一个邻接点,则返回-1 */ArcNode *p;int v1,w1;v1=LocateVex(G,v); /* v1 为顶点v 在图G 中的序号*/w1=LocateVex(G,w); /* w1 为顶点w 在图G 中的序号*/p=G.vertices[v1].firstarc;while(p&&p->adjvex!=w1) /* 指针p 不空且所指表结点不是w */p=p->nextarc;if(!p||!p->nextarc) /* 没找到w 或w 是最后一个邻接点*/return -1;else /* p->adjvex==w */return p->nextarc->adjvex; /* 返回v 的(相对于w 的)下一个邻接顶点的序号*/}Boolean visited[MAX_VERTEX_NUM]; /* 访问标志数组(全局量) */void(*VisitFunc)(char* v); /* 函数变量(全局量) */void DFS(ALGraph G,int v){ /* 从第v 个顶点出发递归地深度优先遍历图G。算法7.5 */int w;VertexType v1,w1;strcpy(v1,*GetVex(G,v));visited[v]=TRUE; /* 设置访问标志为TRUE(已访问) */VisitFunc(G.vertices[v].data); /* 访问第v 个顶点*/for(w=FirstAdjVex(G,v1);w>=0;w=NextAdjVex(G,v1,strcpy(w1,*GetVex(G,w))))if(!visited[w])DFS(G,w); /* 对v 的尚未访问的邻接点w 递归调用DFS */}void DFSTraverse(ALGraph G,void(*Visit)(char*)){ /* 对图G 作深度优先遍历。算法7.4 */int v;VisitFunc=Visit; /* 使用全局变量VisitFunc,使DFS 不必设函数指针参数*/for(v=0;v<G.vexnum;v++)visited[v]=FALSE; /* 访问标志数组初始化*/for(v=0;v<G.vexnum;v++)if(!visited[v])DFS(G,v); /* 对尚未访问的顶点调用DFS */printf(" ");}typedef int QElemType; /* 队列类型*/#include"LinkQueueDef.h"#include"LinkQueueAlgo.h"void BFSTraverse(ALGraph G,void(*Visit)(char*)){/*按广度优先非递归遍历图G。使用辅助队列Q 和访问标志数组visited。算法7.6 */int v,u,w;VertexType u1,w1;LinkQueue Q;for(v=0;v<G.vexnum;++v)visited[v]=FALSE; /* 置初值*/InitQueue(Q); /* 置空的辅助队列Q */for(v=0;v<G.vexnum;v++) /* 如果是连通图,只v=0 就遍历全图*/if(!visited[v]) /* v 尚未访问*/{visited[v]=TRUE;Visit(G.vertices[v].data);EnQueue(Q,v); /* v 入队列*/while(!QueueEmpty(Q)) /* 队列不空*/{DeQueue(Q,u); /* 队头元素出队并置为u */strcpy(u1,*GetVex(G,u));for(w=FirstAdjVex(G,u1);w>=0;w=NextAdjVex(G,u1,strcpy(w1,*GetVex(G,w))))if(!visited[w]) /* w 为u 的尚未访问的邻接顶点*/{visited[w]=TRUE;Visit(G.vertices[w].data);EnQueue(Q,w); /* w 入队*/}}}printf(" ");}void Display(ALGraph G){ /* 输出图的邻接矩阵G */int i;ArcNode *p;switch(G.kind){ case DG: printf("有向图 "); break;case DN: printf("有向网 "); break;case AG: printf("无向图 "); break;case AN: printf("无向网 ");}printf("%d 个顶点: ",G.vexnum);for(i=0;i<G.vexnum;++i)printf("%s ",G.vertices[i].data);printf(" %d 条弧(边): ",G.arcnum);for(i=0;i<G.vexnum;i++){p=G.vertices[i].firstarc;while(p){if(G.kind<=1) /* 有向*/{printf("%s→%s ",G.vertices[i].data,G.vertices[p->adjvex].data);if(G.kind==DN) /* 网*/printf(":%d ",*(p->info));}else /* 无向(避免输出两次) */{if(i<p->adjvex){printf("%s-%s ",G.vertices[i].data,G.vertices[p->adjvex].data);if(G.kind==AN) /* 网*/printf(":%d ",*(p->info));}}p=p->nextarc;}printf(" ");}}/* .........................*//* .........................*/#include "pubuse.h"#define MAX_NAME 3 /* 顶点字符串的最大长度+1 */typedef int InfoType; /* 存放网的权值*/typedef char VertexType[MAX_NAME]; /* 字符串类型*/#include"ALGraphDef.h"#include"ALGraphAlgo.h"void print(char *i){ printf("%s ",i);}void main(){ int i,j,k,n; ALGraph g; VertexType v1,v2; printf("请选择有向图 "); CreateGraph(g); Display(g); printf("深度优先搜索的结果: "); DFSTraverse(g,print); printf("广度优先搜索的结果: "); BFSTraverse(g,print); DestroyGraph(g); /* 销毁图*/}
拓扑排序排课表
信息工程系软件技术学生课程表(拓扑排序)拓扑图为:(图不好粘贴)运用拓扑概念排序的结果:C1 , C9 , C3 , C2 , C7 , C4, C5 , C8 , C6C1计算机应用基础 C2 C语言 C3 VB语言 C4 JSP C5数字逻辑电路 C6软件工程C7计算机网络基础 C8 Java语言 C9计算机数学基础/*-------------------------------主类-----------------------------*/public class Navy1 { public static void main(String[] args) { topology(); //调用拓扑的构造方法 } public static void topology() { //构造拓扑方法 /** 声明拓扑图中的元素 定义节点和节点之间的关系 Entry(a,b)a为b的前导 **/ Entry[] relations = { new Entry(9, 2), new Entry(3,7), new Entry(7, 5), new Entry(5, 8), new Entry(8, 6), new Entry(4, 6), new Entry(1, 3), new Entry(7, 4), new Entry(9, 5), new Entry(2, 8) }; int n = 9; int n1 = 9; /*计算拓扑图中节点数*/ int[] count = { -1, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; /*开辟内存空间*/ Node[] top = { null, null, null, null, null, null, null, null, null, null }; Node p = null; for (int i = 0; i < relations.length; i++) { count[relations[i].k]++; p = new Node(); p.suc = relations[i].k; p.next = top[relations[i].j]; top[relations[i].j] = p; } int r = 0; int[] qlink = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; for (int i = 1; i <= n; i++) { if (count[i] == 0) { qlink[r] = i; r = i; } } int f = qlink[0]; System.out.println("题目及要求:"); System.out.println("课程排课程序。写一个程序,实现对某个专业的课程进行排课的功能。"); System.out.println("已知某专业的课程和它们的前导和后续关系(以有向图的形式表示),"); System.out.println("请用拓扑排序算法求出这些课程的优先关系并输出一种排课结果"); System.out.println("--------------------------------------"); System.out.println("08信息工程系软件技术课程表(拓扑排序)"); while (true) { System.out.println(f); if (f == 0) //结束条件 { break; } else { n1--; p = top[f]; while (true) { if (p == null) { break; } else { count[p.suc]--; if (count[p.suc] == 0) { qlink[r] = p.suc; r = p.suc; } p = p.next; } } f = qlink[f]; } } System.out.println("结束的标志为:" + n1); System.out.println("--------------------------------------------"); System.out.println("注释(数字对应的课程):"); System.out.println("1 计算机应用基础 2 C语言 3 VB语言 "); System.out.println("4 JSP 5 数字逻辑电路 6 软件工程"); System.out.println("7 计算机网络基础 8 Java语言 9 计算机数学基础"); System.out.println("--------------------------------------------"); } /*构造元素类*/ private static class Entry { public Entry(int begin, int end) //定义开始元素和结束元素 { this.j = begin; this.k = end; } int j; int k; } /*声明节点的后继*/ private static class Node { public Node(int suc, Node next) { this.suc = suc; this.next = next; } public Node() { } int suc; Node next; }}
若有向图具有拓扑排序序列,那么它的邻接矩阵必定为?
2->3->1这样顺序的图即可,直接就排除对称和三角了。答案是三角。(这个三角不是特殊矩阵压缩存储时的三角矩阵,而是线性代数中的三角矩阵)可以证明,对于有向图中顶点适当地编号,使其邻接矩阵为三角矩阵且主对角元全为零的充分必要条件是该有向图可以进行拓扑排序。扩展资料:给定有向图G=(VE),并且给定该图G中的任意两个结点u和v,如果结点u与结点v相互可达,即至少存在一条路径可以由结点u开始,到结点v终止,同时存在至少有一条路径可以由结点v开始,到结点u终止,那么就称该有向图G是强连通图。对于有向图最短路问题,计算步骤与求解无向图最短路问题相同,主要区别在于:无向图最短路问题使用单标号法。单标号法是对每一点赋予一个路权标号;而有向最短路问题使用双标号法.双标号法是对每一点赋予两个标号:路径和路权。参考资料来源:百度百科-有向图
为什么拓扑排序不属于内部排序法
由AOV网构造拓扑序列的拓扑排序算法主要是循环执行以下两步,直到不存在入度为0的顶点为止。(1) 选择一个入度为0的顶点并输出之;(2) 从网中删除此顶点及所有出边。循环结束后,若输出的顶点数小于网中的顶点数,则输出“有回路”信息,否则输出的顶点序列就是一种拓扑序列。
已知AOV-网如下图所示:给出该有向图(至少四组)拓扑有序序列,并简述如何进行拓扑排序?
针对该题,可能的拓扑排序:1.c->d->b->a->e2.d->b->c->a->e3.d->c->b->a->e当然,拓扑序列不一定唯一如果图中,这里是aov网中存在有向环,则无法完成拓扑排序。
如果具有N个顶点的有向图能够进行拓扑排序,那么有向
就此题来说,答案应该是一般。另外给出另一方面的分析,希望能对此题有所帮助。题目:若一个有向图具有有序的拓扑排序序列,那么它的邻接矩阵必定为?(比原题加了个有序的)答案是三角。(这个三角不是特殊矩阵压缩存储时的三角矩阵,而是线性代数中的三角矩阵)可以证明,对于有向图中顶点适当地编号,使其邻接矩阵为三角矩阵且主对角元全为零的充分必要条件是该有向图可以进行拓扑排序。
拓扑排序题V={0,1,2,3,4,5,6,7}
こちらの物は お気に召しましたら、持てるだけ お持ち帰りください。こちらの物--这些东西お気に召しましたら--お気に入る的尊敬语,意思是称心,如意,称意;喜爱,喜欢。~たら表示的是假设。这句话的意思是:要是您喜欢的话。持てるだけ お持ち帰りください--持てる是持つ的可能型,后面加一个だけ、表示能尽量拿,这句话是意思是:请您尽管拿,也就是说您能拿多少就拿多少。整句话的意思是:这些东西,您要是喜欢的话,就请您尽管拿吧。
数据结构:利用函数实现图的拓扑排序(高分悬赏)
拓扑排序一.定义对一个有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序(Topological Sort),是将G中所有顶点排成一个线性序列,使得对图中任意一对顶点u和v,若<u,v>∈E(G),则u在线性序列中出现在v之前。通常将这样的线性序列称为满足拓扑次序(Topolgical Order)的序列,简称拓扑序列。二.算法1.无前趋的顶点优先:(1)算法描述:(a)从网中选择一个入度为零的顶点输出;(b)删除该顶点及其于该点有关的所有边;(c)是否还有入度为零的顶点?若有,执行(a),否则结束。算法实现以邻接表为图的存储结构的算法:a)扫描顶点表,将入度为零的顶点入栈;b)当栈非空时:输出栈顶元素v,出栈;检查v的出边,将每条出边的终端顶点的入度减1,若该顶点入度为0,入栈;c)当栈空时,若输出的顶点小于顶点数,则说明AOV网有回路,否则拓扑排序完成。算法实现:void Graph::Toplogicasort()//top是入度为0的顶点栈的栈顶指针 { int top=-1; for(int i=0;i<n;i++) //建立如度为0顶点的链栈 if (count[i]==0) { count[i]=top; top=i; } for(int i=0;i<n;i++) if(top==-1) { cout<<"Network has a cycle"<<endl; return; } else { int j=top;top=count[top];//入度为0的顶点出栈 count<<j<<endl; Edge<float> *l=NodeTable[j].adj; while(l) { int k=l,dest; if(--count[k]==0) { count[k]=top;//入度减至0的顶点进栈 top=k; } l=l->link;//取j的下一条出边 } } } /*通常的拓扑算法要用两个数组,一个用来记录每个顶点的入度,当入度为0,则进栈 。另一个数组用作栈数组,记录入度为0的顶点。其实当入度为0的顶点进栈以后,count[i] =0就不再有用,所以可以用count[i]记录栈内下一个入度为0的顶点,top指向栈顶顶点号 */
若有向图具有拓扑排序序列,那么它的邻接矩阵必定为?
就此题来说,答案应该是一般。另外给出另一方面的分析,希望能对此题有所帮助。题目:若一个有向图具有有序的拓扑排序序列,那么它的邻接矩阵必定为?(比原题加了个有序的)答案是三角。(这个三角不是特殊矩阵压缩存储时的三角矩阵,而是线性代数中的三角矩阵)可以证明,对于有向图中顶点适当地编号,使其邻接矩阵为三角矩阵且主对角元全为零的充分必要条件是该有向图可以进行拓扑排序。
数据结构问题~什么图可以进行拓扑排序~什么图不能进行拓扑排序?
对一个有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序,是将G中所有顶点排成一个线性序列,使得图中任意一对顶点u和v,若<u,v> ∈E(G),则u在线性序列中出现在v之前。拓扑排序(Topological Sort)什么是拓扑序列 通常,这样的线性序列称为满足拓扑次序(Topological Order)的序列,简称拓扑序列。简单的说,由某个集合上的一个偏序得到该集合上的一个全序,这个操作称之为拓扑排序。离散数学中关于偏序和全序的定义: 若集合X上的关系是R是自反的、反对称的和传递的,则称R是集合X上的偏序关系。 设R是集合X上的偏序(Partial Order),如果对每个x,y属于X必有xRy 或 yRx,则称R是集合X上的全序关系。 注意: ①若将图中顶点按拓扑次序排成一行,则图中所有的有向边均是从左指向右的。 ②若图中存在有向环,则不可能使顶点满足拓扑次序。 ③一个DAG的拓扑序列通常表示某种方案切实可行。
拓扑排序是怎么进行的?
由AOV网构造拓扑序列的拓扑排序算法主要是循环执行以下两步,直到不存在入度为0的顶点为止。(1)选择一个入度为0的顶点并输出之;(2)从网中删除此顶点及所有出边。循环结束后,若输出的顶点数小于网中的顶点数,则输出“有回路”信息,否则输出的顶点序列就是一种拓扑序列。
请解释下拓扑排序的定义。。和实现方法。。别复制百度百科。。
拓扑排序 所谓拓扑序列,就是有向图的最长路径问题,如果图中存在环,则最长路径是无法求得的,所以有拓扑序列的有向图不可以存在环。具体定义如下: 给出有向图G=(V,E),若结点的线形序列V1,V2,...Vn满足条件:对于i,j(1≤j<i≤n),Vi和Vj之间没有边。求线形序列V1,V2,...Vn的过程就称为拓扑排序,这个线形序列就称为拓扑序列。【拓扑排序主要思想】 有向图可以拓扑排序的条件是:图中没有环。 具体方法: ⑴ 从图中选择一个入度为0的点加入拓扑序列。 ⑵ 从图中删除该结点以及它的所有出边(即与之相邻点入度减1)。 反复执行这两个步骤,直到所有结点都已经进入拓扑序列。【实例:士兵排队问题】有n个士兵(1≤n≤26),依次编号为A,B,C,...,队列训练时,指挥官要把一些士兵从高到矮排成一行。但现在指挥官不能直接获得每个人的身高信息,只能获得“p1比p2高”这样的比较结果,记作(p1>p2)。例如A>B,B>D,F>D,对应的排队方案有三个:AFBD,FABD,ABFD【输入】 k行,每行a b,表示a>b【输出】 一个可行的排队方案【输入样例】A BB DF D 【输出样例】ABFD
数据结构拓扑排序序列
拓扑排序的方法是,先找到第一个没有被指的,就是C1,加入序列。然后擦掉跟C1有关的边,此时C2和C3都满足没有被指,选一个,比如选C2,加入序列,擦掉和C2有关的边,这个时候可以选C3,C4,C5或C6……如此而已
拓扑排序的流程图
由AOV网构造拓扑序列的拓扑排序算法主要是循环执行以下两步,直到不存在入度为0的顶点为止:选择一个入度为0的顶点并输出之;从网中删除此顶点及所有出边。循环结束后,若输出的顶点数小于网中的顶点数,则输出“有回路”信息,否则输出的顶点序列就是一种拓扑序列。由AOV网构造出拓扑序列的实际意义是:如果按照拓扑序列中的顶点次序,在开始每一项活动时,能够保证它的所有前驱活动都已完成,从而使整个工程顺序进行,不会出现冲突的情况。扩展资料拓扑排序(Topological Sorting)为一个有向无环图(DAG,Directed Acyclic Graph)的所有顶点的线性序列。且该序列必须满足下面两个条件:每个顶点出现且只出现一次。若存在一条从顶点A到顶点B的路径,那么在序列中顶点A出现在顶点B的前面。拓扑排序常用来确定一个依赖关系集中,事物发生的顺序。例如,在日常工作中,可能会将项目拆分成A、B、C、D四个子部分来完成,但A依赖于B和D,C依赖于D。为了计算这个项目进行的顺序,可对这个关系集进行拓扑排序,得出一个线性的序列,则排在前面的任务就是需要先完成的任务。参考资料来源:百度百科-拓扑序列参考资料来源:百度百科-拓扑排序
拓扑排序是怎么进行的?
由AOV网构造拓扑序列的拓扑排序算法主要是循环执行以下两步,直到不存在入度为0的顶点为止。(1) 选择一个入度为0的顶点并输出之;(2) 从网中删除此顶点及所有出边。循环结束后,若输出的顶点数小于网中的顶点数,则输出“有回路”信息,否则输出的顶点序列就是一种拓扑序列。
数据结构题目,关于网线编号拓扑排序的
虽然题目里给出了拓扑排序这样的提示,但我还是难想到与拓扑排序有关的解决方法T^T。不过我脑子里有另外一个算法,不知道对不对:给ans数组赋初值:ans[i]=i;枚举每对交叉,将交叉中编号较大的导线的ans值-1,反之将交叉中编号较小的导线的ans值+1按照ans数组的大小,将对应的 i 输出.
采用邻接矩阵存储结构对有向图进行拓扑排序的算法
lint topsort( ALGraph *G) /*拓扑排序*/{ int i,j,k,top =-1; EdgeNode *ptr; for(i=0;i<G->n;i++) /*入度为0入栈*/ { if(G->adjlist[i].indegree==0) { G->adjlist[i].indegree=top; top=i; } } {if(top==-1) return -1; /*网中有回路*/ j=top;for(i=0;i<G->n;i++) { if(top==-1) return -1;/*AOV网中有回路*/ j=top; top=G->adjlist[top].indegree; /*从栈中退出一个入度为0的顶点并输出*/ printf("->%s",G->adjlist[j].vertex); ptr=G->adjlist[j].firstedge; { k=ptr->adjvex; G->adjlist[k].indegree--; /*修改其他顶点的入度*/ if(G->adjlist[k].indegree==0) /*为0入栈*/ while(ptr!=NULL) { G->adjlist[k].indegree=top; top=k; } ptr=ptr->next; } }}
什么是逆拓扑排序
1、从有向图中选择一个出度为0的顶点输出2、删除1中的顶点,并且删除指向该顶点的全部边3、重复上述两步,直到剩余的图中不存在出度为0的顶点为止如上,得到的序列为逆拓扑序列
怎样实现c语言的拓扑排序啊,谁能帮我写一下代码啊 谢谢啊
#include<stdio.h>#include<stdlib.h>#define Max_VertexNum 100 //顶点个数的宏定义/*邻接表结点*/typedef struct node{ int adjvex; /*邻接点域*/ struct node *next;/*链域*/}EdgeNode;/*顶点结点*/typedef struct vnode{ int vertex; //顶点域 int Degree; //存放入度 EdgeNode *firstedge; //边头指针}VertexNode;typedef VertexNode Adjlist[Max_VertexNum];//定义一个邻接表的数组,用于存放顶点的信息/*图的定义*/typedef struct ALGraph{ Adjlist adjlist; int n,e;//图的顶点和边}Graph;/*为了避免重复检测入度为0的顶点,设置一个栈暂存所有入度为0的顶点*/typedef struct stacknode{ int data; struct stacknode *next;}StackNode;typedef struct{ StackNode *top; //栈顶指针}LinkStack;////////////////////////////////////////////////////////////////////////////////////////////辅助栈的相关定义/*初始化栈*/void InitStack(LinkStack *S){ S->top=NULL;}/*判空*/int IsEmpty(LinkStack *S){ return(S->top==NULL);}/*进栈*/void Push(LinkStack *S,int x){ StackNode *p; p=(StackNode*)malloc(sizeof(StackNode)); p->data=x; p->next=S->top; S->top=p;}/*出栈*/int Pop(LinkStack *S){ int x; StackNode *p=S->top; //保存头指针 if(IsEmpty(S)) { printf("栈为空!"); exit(1); } x=p->data; S->top=p->next; //将栈顶结点从栈上摘下 free(p); return x;}/////////////////////////////////////////////////////////////////////////////////////////////////////*建立一个有向图,其存储结构为邻接表*/void CreateGraph(Graph *G){ int i,j,k; EdgeNode *s; printf("请输入图的顶点数和边数: "); scanf("%d %d",&G->n,&G->e); //建立顶点列表 for(i=1;i<=G->n;i++) { printf("请输入顶点的信息: "); printf(" "); scanf("%d",&G->adjlist[i].vertex); G->adjlist[i].firstedge=NULL; } for(k=0;k<G->e;k++) { printf("请输入边(vi,vj)的顶点序号: "); printf(" "); scanf("%d%d",&i,&j); s=(EdgeNode*)malloc(sizeof(EdgeNode)); //生成边表结点 s->adjvex=j; s->next=G->adjlist[i].firstedge; G->adjlist[i].firstedge=s; //将新建的结点插入到边表的头部 }}//////////////////////////////////////////////////////////////////////////////////////////////////////////*对各个顶点求入度*/void FindInDegree(Graph *G){ int i,j; EdgeNode *p; for(i=1;i<=G->n;i++) { G->adjlist[i].Degree=0; } for(j=1;j<=G->n;j++) { for(p=G->adjlist[j].firstedge;p;p=p->next) { G->adjlist[p->adjvex].Degree++; } /* p=G->adjlist[j].firstedge; while(p!=NULL) { G->adjlist[p->adjvex].Degree++; p=p->next; } */ }}////////////////////////////////////////////////////////////////////////////////////////////////////////////////*进行拓扑有序排列*/int TopoSort(Graph *G){ int i,m,k; int count=0; LinkStack S; FindInDegree(G); InitStack(&S); for(i=1;i<=G->n;i++) { if(!G->adjlist[i].Degree) { Push(&S,i); } } while(!IsEmpty(&S)) { m=Pop(&S); printf("(%d,%d)",m,G->adjlist[m].vertex); //输出i号顶点并计数 count++; for(EdgeNode *p=G->adjlist[m].firstedge;p;p= p->next) { k=p->adjvex; if(!(--G->adjlist[k].Degree)) { Push(&S,k); } } } if(count<G->n) { printf("有回路!"); exit(1); } else { return 0; }}/////////////////////////////////////////////////////////////////////////////////////////////////////////////*输出有向图*/void PrintAdjlist(Graph *G){ int i; EdgeNode *p; for(i=1;i<=G->n;i++) { printf("%d:%d",i,G->adjlist[i].vertex); for(p=G->adjlist[i].firstedge;p;p=p->next) { printf("%d",p->adjvex); } printf(" "); } }///////////////////////////////////////////////////////////////////////////////////////////////////////////////////*主函数*/void main(){Graph G;CreateGraph(&G);PrintAdjlist(&G);TopoSort(&G);}
在用邻接表表示图时,拓扑排序算法时间复杂度为多少
O(n + e)。 对于一个具有n个顶点e条弧的有向图来说,刚开始将入度为0的顶点入栈的时间复杂为O(n),在之后顶点出栈时,入度减1的操作共执行了e次,所以整个算法的时间复杂度为O(n + e)。
拓扑排序结果序列中的第一个节点一定是入度大于0的点?
那个选择题的答案错了,根据拓扑排序的原则,序列的第一个节点一定是没有前驱的,入度等于0的,不然怎么能够拓扑排序排在第一位呢,排在第一位也就是说没有任何前导的结点,入度一定为0了
有回路的有向图能不能完成拓扑排序
显然不能。首先,拓扑排序是指对于一个有向无环图G,将G中所有顶点排成一个线性序列,使得图中任意一对顶点u和v,若<u,v> ∈E(G),则u在线性序列中出现在v之前。那么,假设存在回路,v1,v2,v3,……,vn,v1,则边<v1,v2>∈E(G),故v1在v2之前,类似地,v2在v3之前,……,因此,得出,v1在vn之前。又因为<vn,v1>∈E(G),即vn在v1之前。相互矛盾,所以假设不成立。所以,一个图能够进行拓扑排序的一个必要条件就是图中不存在环。
数据结构中 关于图拓扑排序算法 有个地方不太明白 希望能得到解答
我知道你哪里不明白了,你没看见上面的for循环,1,如果不为0,则不执行if了,但执行for循环。2,执行for循环的目的就是把所有的入度减1,减为0的入栈。
【数据结构】请写出以下AOV网的拓扑排序序列
AOV网的拓扑排序序列BCADFE或BCAFDE
数据结构,这个带权图的拓扑排序什么思路?
由AOV网构造拓扑序列的拓扑排序算法主要是循环执行以下两步,直到不存在入度为0的顶点为止。(1) 选择一个入度为0的顶点并输出之;(2) 从网中删除此顶点及所有出边。就是输出所有箭头都向外指的节点,然后删除与该节点相连的边,一直循环直到节点都输出或不能够找到这样的节点。显然,拓扑排序不一定唯一。题目的一种拓扑排序:S,G,D,A,B,H,E,I,F,C,T
数据结构拓扑排序怎么算?
这道题选择B解题的步骤是根据边集画出图 这道题就四个结点 <1,2>表示有一条从结点1到结点2的有向路径,就是从1可以去2,但是不能从2到1.画的时候都遵循这个规律即可。然后是拓扑的规则首先找到一个只有出没有进的结点。你会发现只有结点1符合要求,那么去掉结点1和与结点1有关系的边,那么就剩下结点2、3、4.这个时候结点2、3没有进来的边,所以任选一个都可以。最后剩下结点四从2。就可以知道顺序了结点1是第一个 然后第二位2,3任选,但绝对不可能是4所以就选B。
拓扑排序和冒泡排序的区别是什么
冒泡排序是一种交换排序方式。设有n个数据依次放在数组元素a(1)至a(n)中,用冒泡法对这n个数据进行递增排序的过程为:先比较a(1)与a(2),若逆序则交换之,接着比较a(2)与a(3),若逆序就交换……依次进行,知道将a(n-1)与a(n)比较交换完,才算完成...
数据结构题,叙述对有环无向图求拓扑排序序列的步骤 (2)写出下图的4个不同的拓扑排序序列麻烦解答,谢谢
(1)设对有向无环图G=<V,E>,求得它的一个拓扑序列为S,过程如下: 初始化S为空,然后每次从G中选取一个入度为0的点v,将v插入到S的尾部,再在G中删除点v,并删除所有以v为弧尾的边(即由v引出去的边),如此循环,直到图G中的V为空集时结束。21 2 3 4 5 6 7 81 3 2 4 5 7 6 83 1 4 2 5 6 7 83 1 2 5 4 7 8 6
有向图g可拓扑排序的判别条件是
有向图g可拓扑排序的判别条件是:每个顶点出现且只出现一次。判别,读音pàn bié,汉语词语,指根据不同点加以区分,辨别。根据不同点加以区分,辨别。示例:清·王夫之《姜斋诗话》卷二:“文章本静业,故曰‘仁者之言蔼如也",学术风俗皆於此判别。”判,汉语常用字(一级字),会意兼形声字,最早见于战国文字。本义指分开、判分;由判分可引申为半、分辨、评断、判决义,用作半时“判”也可写作“牉”,以上义均读为pàn。“判”亦可表不顾、豁出去之义,这种意义上的“判”旧读为pān。判,会意兼形声字。从刀,从半,半亦表音。本义指分开、判分。《广雅·释诂一》:“判,分也。”《左传·庄公三年》:“纪于是乎始判。”杜预注:“判,分也。”由判分可引申为半、分辨、评断、判决义。用作半时,判也可写作“牉”。唐玄应《一切经音义》卷二:“判,古文胖,又作牉。”《周礼·地官·媒氏》:“掌万民之判。”郑玄注:“判,半也,得偶而合。”唐宋官制,以高官兼低职称判。元黄公绍《古今韵会举要·翰韵》:“宰相出典州曰判。”此用其假借义。以上义均读为pàn。
一个有向无环图的拓扑排序序列是唯一的么
AOV网络构造的拓扑序列的拓扑排序算法主要通过以下两个步骤循环,直到没有度为0的顶点,选择度为0的顶点,输出,从网络中删除这个顶点和所有向外的边。在循环结束时,如果输出的顶点数小于网络中的顶点数,则会输出“loop”信息,否则,输出顶点序列就是拓扑序列。利用AOV网络构建拓扑序列的实际意义是:如果按照拓扑序列的顶点顺序,在启动每个活动时,可以保证其所有前体活动都已完成,从而使整个项目有序进行,不会产生冲突。扩展资料:拓扑排序是一个DAG(有向无环图)的所有顶点的线性序列。序列必须满足以下两个条件:每个顶点只出现一次。如果从顶点A到顶点B有一条路径,那么在序列中顶点A出现在顶点B之前。拓扑排序通常用于确定事物在一组依赖项中发生的顺序。例如,在日常工作中,一个项目可能分为A,B,C和D,但取决于B和D和C取决于D计算项目的顺序,关系集可以拓扑排序产生一个线性序列,和第一个任务是需要首先完成。参考资料:百度百科-拓扑序列参考资料:百度百科-拓扑序列
PASCAL拓扑排序~~
拓扑排序是图论的基本算法之一;做一件事情,比如盖新房子,会分成很多工作:买地,买砖,打桩,砌砖,粉刷,放家具....但是这些工作是有一定顺序的,比如说不可能砖都没砌好就开始粉刷,盖房必须在买地买砖都完成后才能开始..所以对于一件事我们总要确定工作顺序,确定这个顺序的过程就是拓扑排序。算法思想是把整个事情看成一个有向无环图,每一个工作是一个点,如点A到B有弧则表示工作B必须在工作A之后完成。(1)将所有入度为0(即没有被任何工作限制的)节点放入堆栈S(2)堆栈中弹出一个节点,把有向图中这个点及它连接的所有边删除(3)跳到(1),直到有向图被全部删除出栈的序列就是所要求的拓扑序列.
求一个可以输出所有拓扑排序的代码或者思路,记住是所有可能的拓扑排序????????????????
伪代码:// 参数g为图,nodes为已经排序的顶点TopologicalSort(Graph g, List nodes) { g中所有没有前驱的顶点放入队列q; while(q不为空) { q中顶点出队,设为v; 复制nodes生成nodes的副本,并把v加入nodes的副本; 复制g生成g的副本,g的副本删去v和v发出的有向边; if (g的副本中没有顶点) { 输出nodes; // node为最终的排序 } else { TopologicalSort(g的副本, nodes的副本);//递归对剩下的图排序 } }}
简单拓扑排序算法C语言
#include <cstdio>#include <cstring>#include <stack>using namespace std;///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// Description: 表示图的结点的邻接边struct Edge{ int dest; Edge *next;} **graph;///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// Description: 添加一个边// Input: e - 要添加边的结点, p - 目的地 // Output: e - 添加边后的结点void AddEdge(Edge *&e, int p){ if(!e) { e = new Edge; e->dest = p; e->next = 0; } else { Edge *tail = e; while (tail->next) tail = tail->next; tail->next = new Edge; tail = tail->next; tail->dest = p; tail->next = 0; }}///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// Description: 输入结点之间的边// Input: Console下用户输入,起点和终点; m - 边的个数// Output: graph - 图;void Input(int &m){ int i, a, b; // a->b存在边(有向) for (i = 0; i < m; i++) { scanf("%d %d", &a, &b); AddEdge(graph[a], b); }}///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// Description: 获得每个结点的入度// Input: n - 结点的个数// Output: degree - 每个结点的入度void GetDegree(int *degree, int n){ int i = 0; Edge *edge; memset(degree, 0, sizeof(int) * n); for (i = 0; i < n; i++) { edge = graph[i]; while(edge) { degree[edge->dest]++; edge = edge->next; } }}///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// Description: 拓扑排序// Input: n - 结点个数// Output: console下输出一个正确的拓扑序void TopoSort(int n){ int *degree = new int[n]; // 初始化所有结点的入度 GetDegree(degree, n); stack<int> s; // 获得入度为0的结点,并入栈 int i = 0; for (i = 0; i < n; i++) if (degree[i] == 0) s.push(i); int index = 0; // 结点的下标 Edge *edge; // 当前结点邻接表 while (!s.empty()) { index = s.top(); printf("%d", index); s.pop(); edge = graph[index]; while (edge) { if (--degree[edge->dest] == 0) s.push(edge->dest); edge = edge->next; } } delete []degree;}int main(){ int n, m; // 结点个数、边个数 scanf("%d %d", &n, &m); int i; graph = new Edge*[n]; for(i = 0; i < n; i++) graph[i] = 0; Input(m); TopoSort(n); return 0;}
拓扑排序 编程
*/#include <stdio.h>#include <malloc.h>// 输出有向图的一个拓扑序列。实现算法7.12的程序 // 图的邻接表存储表示 #define MAX_NAME 3 // 顶点字符串的最大长度+1 #define MAX_VERTEX_NUM 20typedef int InfoType; // 存放网的权值 typedef char VertexType[MAX_NAME]; // 字符串类型 typedef enum{DG,DN,AG,AN}GraphKind; // {有向图,有向网,无向图,无向网} typedef struct ArcNode{ int adjvex; // 该弧所指向的顶点的位置 struct ArcNode *nextarc; // 指向下一条弧的指针 InfoType *info; // 网的权值指针) }ArcNode; // 表结点 typedef struct VNode{ VertexType data; // 顶点信息 ArcNode *firstarc; // 第一个表结点的地址,指向第一条依附该顶点的弧的指针 }VNode,AdjList[MAX_VERTEX_NUM];// 头结点 typedef struct{ AdjList vertices; int vexnum,arcnum; // 图的当前顶点数和弧数 int kind; // 图的种类标志 }ALGraph;// 若G中存在顶点u,则返回该顶点在图中位置;否则返回-1。 int LocateVex(ALGraph G,VertexType u){ int i; for(i=0;i<G.vexnum;++i) if(strcmp(u,G.vertices[i].data)==0) return i; return -1;}// 采用邻接表存储结构,构造没有相关信息的图G(用一个函数构造4种图)。int CreateGraph(ALGraph *G){ int i,j,k; int w; // 权值 VertexType va,vb; ArcNode *p; printf("请输入图的类型(有向图:0,有向网:1,无向图:2,无向网:3): "); scanf("%d",&(*G).kind); printf("请输入图的顶点数和边数:(空格) "); scanf("%d%d", &(*G).vexnum, &(*G).arcnum); printf("请输入%d个顶点的值(<%d个字符): ",(*G).vexnum,MAX_NAME); for(i = 0; i < (*G).vexnum; ++i) // 构造顶点向量 { scanf("%s", (*G).vertices[i].data); (*G).vertices[i].firstarc = NULL; } if((*G).kind == 1 || (*G).kind == 3) // 网 printf("请顺序输入每条弧(边)的权值、弧尾和弧头(以空格作为间隔): "); else // 图 printf("请顺序输入每条弧(边)的弧尾和弧头(以空格作为间隔): "); for(k = 0;k < (*G).arcnum; ++k) // 构造表结点链表 { if((*G).kind==1||(*G).kind==3) // 网 scanf("%d%s%s",&w,va,vb); else // 图 scanf("%s%s",va,vb); i = LocateVex(*G,va); // 弧尾 j = LocateVex(*G,vb); // 弧头 p = (ArcNode*)malloc(sizeof(ArcNode)); p->adjvex = j; if((*G).kind == 1 || (*G).kind == 3) // 网 { p->info = (int *)malloc(sizeof(int)); *(p->info) = w; } else p->info = NULL; // 图 p->nextarc = (*G).vertices[i].firstarc; // 插在表头 (*G).vertices[i].firstarc = p; if((*G).kind >= 2) // 无向图或网,产生第二个表结点 { p = (ArcNode*)malloc(sizeof(ArcNode)); p->adjvex = i; if((*G).kind == 3) // 无向网 { p->info = (int*)malloc(sizeof(int)); *(p->info) = w; } else p->info = NULL; // 无向图 p->nextarc = (*G).vertices[j].firstarc; // 插在表头 (*G).vertices[j].firstarc = p; } } return 1;}// 输出图的邻接表G。void Display(ALGraph G){ int i; ArcNode *p; switch(G.kind) { case DG: printf("有向图 "); break; case DN: printf("有向网 "); break; case AG: printf("无向图 "); break; case AN: printf("无向网 "); } printf("%d个顶点: ",G.vexnum); for(i = 0; i < G.vexnum; ++i) printf("%s ",G.vertices[i].data); printf(" %d条弧(边): ", G.arcnum); for(i = 0; i < G.vexnum; i++) { p = G.vertices[i].firstarc; while(p) { if(G.kind <= 1) // 有向 { printf("%s→%s ",G.vertices[i].data, G.vertices[p->adjvex].data); if(G.kind == DN) // 网 printf(":%d ", *(p->info)); } else // 无向(避免输出两次) { if(i < p->adjvex) { printf("%s-%s ",G.vertices[i].data, G.vertices[p->adjvex].data); if(G.kind == AN) // 网 printf(":%d ",*(p->info)); } } p=p->nextarc; } printf(" "); }}// 求顶点的入度,算法7.12、7.13调用void FindInDegree(ALGraph G,int indegree[]){ int i; ArcNode *p; for(i=0;i<G.vexnum;i++) indegree[i]=0; // 赋初值 for(i=0;i<G.vexnum;i++) { p=G.vertices[i].firstarc; while(p) { indegree[p->adjvex]++; p=p->nextarc; } }}typedef int SElemType; // 栈类型 #define STACK_INIT_SIZE 10 // 存储空间初始分配量 #define STACKINCREMENT 2 // 存储空间分配增量 // 栈的顺序存储表示 P46 typedef struct SqStack{ SElemType *base; // 在栈构造之前和销毁之后,base的值为NULL SElemType *top; // 栈顶指针 int stacksize; // 当前已分配的存储空间,以元素为单位 }SqStack; // 顺序栈// 构造一个空栈S。int InitStack(SqStack *S){ // 为栈底分配一个指定大小的存储空间 (*S).base = (SElemType *)malloc(STACK_INIT_SIZE*sizeof(SElemType)); if( !(*S).base ) exit(0); // 存储分配失败 (*S).top = (*S).base; // 栈底与栈顶相同表示一个空栈 (*S).stacksize = STACK_INIT_SIZE; return 1;}// 若栈S为空栈(栈顶与栈底相同的),则返回1,否则返回0。int StackEmpty(SqStack S){ if(S.top == S.base) return 1; else return 0;}// 插入元素e为新的栈顶元素。int Push(SqStack *S, SElemType e){ if((*S).top - (*S).base >= (*S).stacksize) // 栈满,追加存储空间 { (*S).base = (SElemType *)realloc((*S).base, ((*S).stacksize + STACKINCREMENT) * sizeof(SElemType)); if( !(*S).base ) exit(0); // 存储分配失败 (*S).top = (*S).base+(*S).stacksize; (*S).stacksize += STACKINCREMENT; } *((*S).top)++=e; // 这个等式的++ * 优先级相同,但是它们的运算方式,是自右向左 return 1;}// 若栈不空,则删除S的栈顶元素,用e返回其值,并返回1;否则返回0。int Pop(SqStack *S,SElemType *e){ if((*S).top == (*S).base) return 0; *e = *--(*S).top; // 这个等式的++ * 优先级相同,但是它们的运算方式,是自右向左 return 1;}// 算法7.12 P182// 有向图G采用邻接表存储结构。若G无回路,则输出G的顶点的一个拓扑序// 列并返回1, 否则返回0。int TopologicalSort(ALGraph G){ int i,k,count,indegree[MAX_VERTEX_NUM]; SqStack S; ArcNode *p; FindInDegree(G,indegree); // 对各顶点求入度indegree[0..vernum-1] InitStack(&S); // 初始化栈 for(i=0;i<G.vexnum;++i) // 建零入度顶点栈S if(!indegree[i]) Push(&S,i); // 入度为0者进栈 count=0; // 对输出顶点计数 while(!StackEmpty(S)) { // 栈不空 Pop(&S,&i); printf("%s ",G.vertices[i].data); // 输出i号顶点并计数 ++count; for(p=G.vertices[i].firstarc;p;p=p->nextarc) { // 对i号顶点的每个邻接点的入度减1 k=p->adjvex; if(!(--indegree[k])) // 若入度减为0,则入栈 Push(&S,k); } } if(count<G.vexnum) { printf("此有向图有回路 "); return 0; } else { printf("为一个拓扑序列。 "); return 1; }}int main(){ ALGraph f; printf("请选择有向图 "); CreateGraph(&f); Display(f); TopologicalSort(f); system("pause"); return 0; }/*输出效果:请选择有向图请输入图的类型(有向图:0,有向网:1,无向图:2,无向网:3): 0请输入图的顶点数和边数:(空格)4 4请输入4个顶点的值(<3个字符):abcd请顺序输入每条弧(边)的弧尾和弧头(以空格作为间隔):a ba cb dc d有向图4个顶点:a b c d4条弧(边):a→c a→bb→dc→da b c d 为一个拓扑序列。请按任意键继续. . . */
利用拓扑排序可以判断有向图是否存在环
拓扑排序的核心就是每次找入度为0的点 进入输出队列 然后将与此点相连的节点入度减1 重复做当做n-1 次后还有点没进输出队列 那么这些点就是环上的 因为环上的各点入度都为1 没有0的 就不能更新
求拓扑排序算法的详细讲解
3.1AOV网 在现代化管理中,人们常用有向图来描述和分析一项工程的计划和实施过程,一个工程常被分为多个小的子工程,这些子工程被称为活动(Activity),在有向图中若以顶点表示活动,有向边表示活动之间的先后关系,这样的图简称为AOV网。如下图是计算机专业课程之间的先后关系:基础知识课程应先于其它所有课程,pascal语言课程应先于数据结构。3. 2 拓扑排序 在AOV网中为了更好地完成工程,必须满足活动之间先后关系,需要将各活动排一个先后次序即为拓扑排序。如上图的拓扑排序基础知识;Pascal;数据结构;离散数学。或基础知识;离散数学Pascal;数据结构。拓扑排序的方法和步骤:(1)在图中选一个没有前趋的顶点并输出之(2)删除该顶点及由它发出的各边,直到图中不存在没有前趋的顶点为止。若图中存在回路,拓扑排序无法进行。以下是将一AOV网进行拓扑排序的算法:网采用邻接矩阵A表示,若a[i,j]=1,表示活动i先于j,a[i,j]=0,表示活动i与j不存在先后关系。(1)计算各顶点的入度 (2)找入度为零的点输出之,删除该点,且与该点关联各点的入度减1 (3)若所有顶点都输出完毕。 程序如下: program tppv;const maxn=100;var map:array[1..maxn,1..maxn] of byte; into:array[1..maxn] of byte; n,i,j,k:byte;procedure init;var i,j:integer;begin read(n); for i:=1 to n do for j:=1 to n do begin read(map[i,j]); inc(into[j]); end;end;begin init; for i:=1 to n do begin j:=1; while (j<=n)and(into[j]<>0) do inc(j); write(j," "); into[j]:=255; for k:=1 to n do if map[j,k]=1 then dec(into[k]); end;end. 3.3应用举例与练习 例:士兵排队问题: 有N个士兵(1<=N<=100),编号依次为1,2,...,N.队列训练时,指挥官要把士兵从高到矮排成一行,但指挥官只知道“1 比2 高,7 比 5高”这样的比较结果。输入文件:第一行为数N(N〈=100);表示士兵的个数。以下若干行每行两个数A,B 表示A高于B。输出文件:给出一个合法的排队序列。程序如下:program tppv;const maxn=100;var map:array[1..maxn,1..maxn] of byte; into:array[1..maxn] of byte; n,i,j,k:byte; fp:text;procedure init;var i,j:integer;begin assign(fp,"tp.txt"); reset(fp); readln(fp,n); fillchar(map,sizeof(map),0); fillchar(into,sizeof(into),0); while not(seekeof(fp)) do begin readln(fp,i,j); map[i,j]=1 ; inc(into[j]); end; close(fp);end;begin init; for i:=1 to n do begin j:=1; while (j<=n)and(into[j]<>0) do inc(j); write(j," "); into[j]:=255; for k:=1 to n do if map[j,k]=1 then dec(into[k]); end;end.练习:Z语言问题:Z语言的基本字母也是26个,不妨用a到z表示,但先后顺序与英语不同。现在按Z语言的顺序给出了N个单词,请依照这些单词给出一个可能的Z语言字母顺序。输入文件:第一行一个数N(N<=100)表示单词的个数。 第二行到第N+1行,每行一个单词。输出文件:仅一行,可能的字母顺序。(图不好粘贴)
有环有向图究竟可以拓扑排序吗???
有环有向图 在工程上是不可能实际存在的,因为不可能自己即是自己的先导又是自己的后驱 比如, 某学校规定,修完离散数学才能修数据结构,修完数据结构以后的下一门课程一定不是离散数学了。 这里的修完是指学完并考试通过,所以是重修的话还是进行不了下一道工序。 修完数据结构才能修离散,但修完理论就得修数据结构,这显然是一个死循环 所以有环有向图进行拓扑排序是一个伪概念,严蔚敏书上有讲的 假设你有这样一个环,那么理论是可能让你一直在这里环里面做死循环的。 [ ]
写出三种拓扑排序
如图
怎样通过编程实现拓扑排序
实现的基本方法拓扑排序方法如下:(1)从有向图中选择一个没有前驱(即入度为0)的顶点并且输出它.(2)从网中删去该顶点,并且删去从该顶点发出的全部有向边.(3)重复上述两步,直到剩余的网中不再存在没有前驱的顶点为止.拓扑序列C++核心代码boolTopologicalSort(inta[][101])//可以完成拓扑排序则返回True{intn=a[0][0],i,j;intinto[101],ans[101];memset(into,0,sizeof(into));memset(ans,0,sizeof(ans));for(i=1;i<=n;i++){for(j=1;j<=n;j++){if(a[i][j]>0)into[j]++;}}into[0]=1;for(i=1;i<=n;i++){j=0;while(into[j]!=0){j++;if(j>n)returnfalse;}ans[i]=j;into[j]=-1;for(intk=1;k<=n;k++){if(a[j][k]>0)into[k]--;}}for(i=1;i<=n;i++){cout<<ans[i]<<"";}cout<<endl;returntrue;}
终点序号从小到大的次序写拓扑排序是什么意思
终点序号从小到大的次序写拓扑排序是先后排序的意思。根据查询相关公开信息显示,终点序号从小到大的次序写拓扑排序一个不确定的偏序关系经全序后就有一种确定的先后顺序了,拓扑排序就是对有向无环图(DAG)上的一些点的编号进行排序,使得图中每条边的起点都比终点的位置要靠前。
数据结构中有关拓扑排序的相关知识
将入度为0的结点入队,删除后同时将所有相邻顶点的先决条件减一。当某个顶点的计数为0时,将它入队。这是关键思想。
拓扑排序一定是三角矩阵吗
拓扑排序一定是三角矩阵。上三角矩阵指的主对角线下方的元素全为零,而对角矩阵指的是主对角线上方与下方的元素都为零。所以对角阵一定是上三角阵,但上三角阵不一定是对角阵。可以证明,对于有向图中顶点适当地编号,使其邻接矩阵为三角矩阵且主对角元全为零的充分必要条件是该有向图可以进行拓扑排序。非计算机应用:拓扑排序常用来确定一个依赖关系集中,事物发生的顺序。例如,在日常工作中,可能会将项目拆分成A、B、C、D四个子部分来完成,但A依赖于B和D,C依赖于D。为了计算这个项目进行的顺序,可对这个关系集进行拓扑排序,得出一个线性的序列,则排在前面的任务就是需要先完成的任务。注意:这里得到的排序并不是唯一的!就好像你早上穿衣服可以先穿上衣也可以先穿裤子,只要里面的衣服在外面的衣服之前穿就行。
拓扑排序时间复杂度o(n+e)怎么算的?
对有n个顶点和e条弧的有向图而言,建立求各顶点的入度的时间复杂度为O(e);建零入度顶点栈的时间复杂度为O(n);在拓扑排序过程中,若有向图无环,则每个顶点进一次栈、出一次栈,入度减1的操作在while语句中总共执行e次,所以总的时间复杂度为O(n+e)。对一个有向无环图(Directed Acyclic Graph简称DAG)G进行拓扑排序,是将G中所有顶点排成一个线性序列,使得图中任意一对顶点u和v,若边(u,v)∈E(G),则u在线性序列中出现在v之前。通常,这样的线性序列称为满足拓扑次序(Topological Order)的序列,简称拓扑序列。简单的说,由某个集合上的一个偏序得到该集合上的一个全序,这个操作称之为拓扑排序。时间复杂度是同一问题可用不同算法解决,而一个算法的质量优劣将影响到算法乃至程序的效率。算法分析的目的在于选择合适算法和改进算法。计算机科学中,算法的时间复杂度是一个函数,它定性描述了该算法的运行时间。这是一个关于代表算法输入值的字符串的长度的函数。时间复杂度常用大O符号表述,不包括这个函数的低阶项和首项系数。使用这种方式时,时间复杂度可被称为是渐近的,它考察当输入值大小趋近无穷时的情况。扩展资料在计算时间复杂度的时候,先找出算法的基本操作,然后根据相应的各语句确定它的执行次数,再找出 T(n) 的同数量级,找出后,f(n) = 该数量级,若 T(n)/f(n) 求极限可得到一常数c,则时间复杂度T(n) = O(f(n))。按数量级递增排列,常见的时间复杂度有:常数阶O(1),线性阶O(n),线性对数阶O(nlog2n),平方阶O(n^2),立方阶O(n^3),...,k次方阶O(n^k),指数阶O(2^n)。随着问题规模n的不断增大,上述时间复杂度不断增大,算法的执行效率越低。参考资料来源:百度百科-拓扑排序参考资料来源:百度百科-时间复杂度