拓扑排序

拓扑排序

大家好,又见面了,我是全栈君,祝每个程序员都可以多学几门语言。

拓扑排序是对有向无环图的一种排序。表示了顶点按边的方向出现的先后顺序。假设有环,则无法表示两个顶点的先后顺序。

在现实生活中,也会有不少应用样例,比方学校课程布置图,要先修完一些基础课,才干够继续修专业课。
一个简单的求拓扑排序的算法:首先要找到随意入度为0的一个顶点,删除它及全部相邻的边,再找入度为0的顶点,以此类推,直到删除全部顶点。顶点的删除顺序即为拓扑排序。
    非常easy得到拓扑排序的伪代码:
    void TopSort(Graph g)
    {
        for (int i=0; i<vertexnum; i++)
        {
            vertex v = FindZeroIndegree(g);
            if (v is not vertex)        
                cout <<“the graph has cycle”<<endl;
            cout << v ;
            foreach vertex w adjacent to v
                w.indegree–;
        }
    }
拓扑排序   
    相同以上图为例,对于该图进行拓扑排序会得到:v1 v2 v5 v4 v3 v7 v6 或者v1 v2 v5 v4 v7 v3 v6 。
    仍然利用上一贴图的构建方法,进行验证。
    代码实现:
 

 
#include <iostream>

using
namespace
std;

#define MAX_VERTEX_NUM    20

struct
adjVertexNode

{

   
int
adjVertexPosition;

   
adjVertexNode
*
next;

};

struct
VertexNode

{

   
char
data
[
2
];

   
adjVertexNode
*
list;

   
int
indegree;

};

struct
Graph

{

   
VertexNode
VertexNode
[
MAX_VERTEX_NUM
];

   
int
vertexNum;

   
int
edgeNum;

};

void
CreateGraph (
Graph
&
g)

{

    
int
i
,
j
,
edgeStart
,
edgeEnd;

    
adjVertexNode
*
adjNode;

    
cout
<<
“Please input vertex and edge num (vnum enum):”
<<
endl;

    
cin
>>
g
.
vertexNum
>>
g
.
edgeNum;

    
cout
<<
“Please input vertex information (v1)
/n
note: every vertex info end with Enter”
<<
endl;

    
for (
i
=
0;
i
<
g
.
vertexNum;
i
++)

    
{

        
cin
>>
g
.
VertexNode
[
i
].
data;
// vertex data info.

        
g
.
VertexNode
[
i
].
list
=
NULL;

        
g
.
VertexNode
[
i
].
indegree
=
0;

    
}

    
cout
<<
“input edge information(start end):”
<<
endl;

    
for (
j
=
0;
j
<
g
.
edgeNum;
j
++)

    
{

        
cin
>>
edgeStart
>>
edgeEnd;

        
adjNode
=
new
adjVertexNode;

        
adjNode
->
adjVertexPosition
=
edgeEnd

1;
// because array begin from 0, so it is j-1

        
adjNode
->
next
=
g
.
VertexNode
[
edgeStart

1
].
list;

        
g
.
VertexNode
[
edgeStart

1
].
list
=
adjNode;

        
//每添加�一条边,则边的End顶点的入度加1

        
g
.
VertexNode
[
edgeEnd

1
].
indegree
++;

    
}

}

void
PrintAdjList(
const
Graph
&
g)

{

   
cout
<<
“The adjacent list for graph is:”
<<
endl;

   
for (
int
i
=
0;
i
<
g
.
vertexNum;
i
++)

   
{

       
cout
<<
g
.
VertexNode
[
i
].
data
<<
“->”;

       
adjVertexNode
*
head
=
g
.
VertexNode
[
i
].
list;

       
if (
head
==
NULL)

           
cout
<<
“NULL”;

       
while (
head
!=
NULL)

       
{

           
cout
<<
head
->
adjVertexPosition
+
1
<<
” “;

           
head
=
head
->
next;

       
}

       
cout
<<
endl;

   
}

}

VertexNode
&
FindZeroIndegree(
Graph
&
g)

{

   
for (
int
i
=
0;
i
<
g
.
vertexNum;
i
++)

   
{

       
if (
g
.
VertexNode
[
i
].
indegree
==
0)

           
return
g
.
VertexNode
[
i
];

   
}

   
return
g
.
VertexNode
[
0
];

}

void
TopSort(
Graph
&
g)

{

   
cout
<<
“The topsort is:”
<<
endl;

   
for (
int
i
=
0;
i
<
g
.
vertexNum;
i
++)

   
{

       
VertexNode
&
v
=
FindZeroIndegree(
g);

       
if (
v
.
indegree
!=
NULL)

           
cout
<<
“The graph has cycle, can not do topsort”
<<
endl;

       
// print graph as topsort.

       
cout
<<
v
.
data
<<
” “;

       
// for each vertex w adjacent to v, –indegree

       
adjVertexNode
*
padjv
=
v
.
list;

       
while (
padjv
!=
NULL)

       
{
//!!这个算法这里破坏了原图中的入度信息。最后入度均为1

           
g
.
VertexNode
[
padjv
->
adjVertexPosition
].
indegree
;

           
padjv
=
padjv
->
next;

       
}

       
//避免入度信息均为零FindZeroIndegree找到删除的顶点,将删除的顶点入度置为1

       
v
.
indegree
++;

   
}

   
cout
<<
endl;

}

void
DeleteGraph(
Graph
&
g)

{

   
for (
int
i
=
0;
i
<
g
.
vertexNum;
i
++)

   
{

       
adjVertexNode
*
tmp
=
NULL;

       
while(
g
.
VertexNode
[
i
].
list
!=
NULL)

       
{

           
tmp
=
g
.
VertexNode
[
i
].
list;

           
g
.
VertexNode
[
i
].
list
=
g
.
VertexNode
[
i
].
list
->
next;

           
delete
tmp;

           
tmp
=
NULL;

       
}

   
}

}

int
main(
int
argc
,
const
char
**
argv)

{

   
Graph
g;

   
CreateGraph(
g);

   
PrintAdjList(
g);

   
TopSort(
g);

   
DeleteGraph(
g);

   
return
0;

}

 执行结果:
拓扑排序
 

     从上面的代码能发现
FindZeroIndegree的时间复杂度为O(|V|),TopSort的时间复杂度为O(|V|2)
     原因在于,每次删除顶点,仅仅有邻接点须要调整入度,但
FindZeroIndegree却是遍历了全部顶点,甚至已经删除的顶点。
     更为合理的方法是将每次遍历得出的入度为0的顶点放入一个队列。
void
TopSort2(
Graph
&
g)

{

   
queue
<
VertexNode
>
q;

   
for (
int
i
=
0;
i
<
g
.
vertexNum;
i
++)

   
{

       
if (
g
.
VertexNode
[
i
].
indegree
==
0)

           
q
.
push(
g
.
VertexNode
[
i
]);

   
}

   
int
count
=
0;

   
cout
<<
“The topsort is:”
<<
endl;

   
while (
!
q
.
empty())

   
{

       
VertexNode
v
=
q
.
front();

       
q
.
pop();

       
cout
<<
v
.
data
<<
” “;

       
count
++;

       
adjVertexNode
*
padjv
=
v
.
list;

       
while (
padjv
!=
NULL)

       
{
//!!这个算法这里破坏了原图中的入度信息。最后入度均为1

           
if (
(
g
.
VertexNode
[
padjv
->
adjVertexPosition
].
indegree)
==
0)

               
q
.
push(
g
.
VertexNode
[
padjv
->
adjVertexPosition
]);

           
padjv
=
padjv
->
next;

       
}

   
}

   
if (
count
!=
g
.
vertexNum)

       
cout
<<
“The graph has cycle, can not do topsort”
<<
endl;

}
     内部的while循环最多运行|E|次,即每条边运行一次。队列对每一个顶点最多运行一次操作,所以新算法的时间复杂度为O(|E|+|V|). 优于O(|V|2)由于拓扑图边数最多有n(n-1)/2,即O(|E|+|V|)<=O(|V|2)

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/118755.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Redis和MySQL的区别与使用(redis做mysql的缓存并且数据同步)

    Redis和MySQL的区别与使用(redis做mysql的缓存并且数据同步)一、redis和mysql介绍Redis基于内存,读写速度快,也可做持久化,但是内存空间有限,当数据量超过内存空间时,需扩充内存,但内存价格贵。MySQL基于磁盘,读写速度没有Redis快,但是不受空间容量限制,性价比高。大多数的应用场景是MySQL(主)+Redis(辅),MySQL做为主存储,Redis用于缓存,加快访问速度。需要高性能的地方使用Redis,不需要高性能的地方使用M…

    2022年6月15日
    37
  • c++钩子函数(react钩子函数)

    结合自定义消息映射方面,作为学习的一个总结.Step1:创建win32动态链接库(anemptyprojectDLL),命名为HookDLL1:增加一个类,命名为DLL.cpp2:增加头文件#include&lt;windows.h&gt;#include"DLL.h"3:创建全局变量#pragmadata_seg("Shared")HHOOKmHoo…

    2022年4月12日
    46
  • OSError: cannot open resource 错误原因及解决方法[通俗易懂]

    OSError: cannot open resource 错误原因及解决方法[通俗易懂]错误类型最近学习python的时侯,初步学习了wordcloud库,很强大、很易用的一个库。但是我在进行练习的时候却遇到了OSError:cannotopenresource这样的错误,于是逐步分析每行代码,还是找到了错误。问题代码|importwordcloud,jiebaf=open(‘F:/PythonStudy/新时代中国特色社会主义.txt’,’r’,encodin…

    2022年6月24日
    31
  • delphi2007中如何安装ActiveX控件

    delphi2007中如何安装ActiveX控件(1).打开Delphi2007,File-New-PackageDelphiforwin32.(2).Component-ImportComponent,选择ImportActiveXControl.(3).在控件列表,或Add添加选择相应Active控件后,点next.(4).选择安装ActiveX控件所在的面板页,单元,包等.(5)点next,最后一步,选

    2022年5月15日
    39
  • Oracle 视图索引

    Oracle 视图索引第五章视图索引的操作5.1视图的功能一个视图实际上就是封装了一条复杂的查询语句注:为了在当前用户模式中创建视图,要求数据库用户必须有createanyview(创建任何视图)的权限。5.2创建视图的语法create[orreplace]view视图名称as查询语句例:建立一个视图,包含全部部门编号为20的部门的雇员信息(雇员编号,姓名,工作,部门编号)createview…

    2022年7月22日
    10
  • 死锁的四个必要条件和解决办法_半暖的博客_活锁和死锁的概念

    死锁的四个必要条件和解决办法_半暖的博客_活锁和死锁的概念死锁概念及产生原理   概念:多个并发进程因争夺系统资源而产生相互等待的现象。   原理:当一组进程中的每个进程都在等待某个事件发生,而只有这组进程中的其他进程才能触发该事件,这就称这组进程发生了死锁。   本质原因:     1)、系统资源有限。     2)、进程推进顺序不合理。死锁产生的4个必要条件  1、互斥:某种资源一次只允许一个进程访问,即该资源一旦分配给某个进程…

    2022年4月20日
    64

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号