2016/5/19 17:39:07

拓扑排序,是对有向无环图(Directed Acylic Graph , DAG )进行的一种操作,这种操作是将DAG中的所有顶点排成一个线性序列,使得图中的任意一对顶点u,v满足如下条件:

若边(u,v)∈E(G),则在最终的线性序列中出现在v的前面

好了,说人话:拓扑排序的应用常常和AOV网相联系,在一个大型的工程中,某些项目不是独立于其他项目的,这意味着这种非独立的项目的完成必须依赖与其它项目的完成而完成,不妨记为u,v,则若边(u,v)∈E(G),代表着必须在项目u完成后,v才能完成。

这样,拓扑排序就可以表示一个工程的进度安排,这也样也更加方便我们理解为什么一个图如果有环,就一定不存在拓扑排序:因为这相当于在工程中你不停的重复做同一个项目,工程变成一个项目的循环,自然不存在拓扑排序。

当然,拓扑排序往往不会只有一种,通过DFS,我们可以求得拓扑排序。

拓扑排序的思路简述如下:

  1. 状态标记:共三种,-1表示访问中,0表示未访问,1表示已访问,由数组c保存
  2. dfs终止的判别条件:如果存在环,则不存在,退出;反之把当前结点加入拓扑排序的首部(线性序列的当前第一个位置,随着排序的进行,这个位置会不断前移)
  3. 通过topo数组记录拓扑排序

这里解释一下书上的问题:为什么访问完一个节点就把当前结点加入到拓扑排序首部?

  答:因为由拓扑排序的性质可知,在DAG中,不妨任取从u顶点出发进行DFS,遇到v顶点,在最终的拓扑排序中始终应满足u在v之前,而根据DFS满足栈的FIFO性质可知,顶点v会先进入拓扑序列,顶点u后进入拓扑序列,因此,如果我们想要顺序获取拓扑序列,就应该将当前顶点(u)加入到拓扑排序的首部。当然,我们也可以通过模拟栈的FIFO特性,通过弹栈将逆序的拓扑序列变为顺序,但是这样无疑增加的操作的步骤,本质上是一样的。

接下来给出拓扑排序的代码:(图通过邻接矩阵存储)

 #include<cstdio>
#include<cstring>
const int maxn=;
int n,m,u,v,t,topo[maxn],G[maxn][maxn],c[maxn];
//DFS
bool dfs(int u){
c[u]=-;//正在访问中
for(int i=;i<n;i++)if(G[u][i]){
if(c[i]<)return false;//存在环,退出
if(!c[i]&&!dfs(i))return false;//i没访问过且访问后发现有环,退出
}
c[u]=;topo[--t]=u;//u顶点访问完毕,修改状态,加入拓扑序列首部
return true;
}
//topological ordering
bool toposort(){
t=n;
memset(c,,sizeof(c));
for(int i=;i<n;i++)
if(!dfs(i))return false;
return true;
}

好了,介绍完拓扑排序,照例是大餐,算法和OJ搭配食用,味道更棒哦~

传送门:

UVa:10305 Ordering Tasks   https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=838&page=show_problem&problem=1246

这是一道非常简单的拓扑排序,由题意我们可以知道,我们无需判断环的存在,因此代码简洁了不少,不过,题目中要求m和n同时为0时代表结束,一开始写成了m&&n,导致wa,后来发现应该是m||n

另外,在解题时,我用了栈来简化逆序操作,不然我还要计算一下输出序列的长度,再把dfs的结果加入到拓扑排序首部,想想就觉得很麻烦,还不如自己写一个栈来的简单~~~

几个地方在提交的时候要注意:

  1. index作为全局变量,在本地编译可过,UVa上会complication error,所以这里改用了pos来指代栈指针
  2. 注意这是多输入的题目,所以vis数组一定要用memset清空
  3. 本来想把结果输出的printf整合为一句话,但是发现这样的话,换行符无法输出,我想过一段时间找点资料,了解一下printf的实现,就应该能明白问题的根源了。

代码如下:

 #include<cstdio>
#include<cstring>
#define N 100
int g[N+][N+],u,v,n,m,stack[N+],vis[N+],pos=;
void push(int x){
stack[pos++]=x;
}
int pop(){
return stack[--pos];
}
void dfs(int u){
vis[u]=-;//visting
for(int i=;i<=n;i++)if(g[u][i]&&!vis[i])
dfs(i);
push(u);vis[u]=;//visted
}
int main(){
while(scanf("%d%d",&n,&m)==&&(m||n)){//这里是m或n,与的话会wa,我离散数学要去面壁
while(m--){
scanf("%d%d",&u,&v);
g[u][v]=;
}
memset(vis,,sizeof(vis));//initialization
for(int i=;i<=n;i++)//topological sort
if(!vis[i])dfs(i);
while(pos){//print result
printf("%d",pop());
printf("%c",pos>?' ':'\n');//三目运算符真的很好用
}
}
}

这一篇就到这里啦~~~

拓扑排序(Topological Order)UVa10305 Ordering Tasks的更多相关文章

  1. 拓扑排序 (Topological Sorting)

    拓扑排序(Topological Sorting) 一.拓扑排序 含义 构造AOV网络全部顶点的拓扑有序序列的运算称为拓扑排序(Topological Sorting). 在图论中,拓扑排序(Topo ...

  2. UVA-10305 Ordering Tasks (拓扑排序)

    题目大意:给出n个点,m条关系,按关系的从小到大排序. 题目分析:拓扑排序的模板题,套模板. kahn算法: 伪代码: Kahn算法: 摘一段维基百科上关于Kahn算法的伪码描述: L← Empty ...

  3. UVA10305 Ordering Tasks (拓扑序列)

    本文链接:http://www.cnblogs.com/Ash-ly/p/5398586.html 题意: 假设有N个变量,还有M个二元组(u, v),分别表示变量u 小于 v.那么.所有变量从小到大 ...

  4. LeetCode编程训练 - 拓扑排序(Topological Sort)

    拓扑排序基础 拓扑排序用于解决有向无环图(DAG,Directed Acyclic Graph)按依赖关系排线性序列问题,直白地说解决这样的问题:有一组数据,其中一些数据依赖其他,问能否按依赖关系排序 ...

  5. 算法与数据结构基础 - 拓扑排序(Topological Sort)

    拓扑排序基础 拓扑排序用于解决有向无环图(DAG,Directed Acyclic Graph)按依赖关系排线性序列问题,直白地说解决这样的问题:有一组数据,其中一些数据依赖其他,问能否按依赖关系排序 ...

  6. 拓扑排序 Topological Sort

    2018-05-02 16:26:07 在计算机科学领域,有向图的拓扑排序或拓扑排序是其顶点的线性排序,使得对于从顶点u到顶点v的每个有向边uv,u在排序中都在v前.例如,图形的顶点可以表示要执行的任 ...

  7. 【数据结构与算法Python版学习笔记】图——拓扑排序 Topological Sort

    概念 很多问题都可转化为图, 利用图算法解决 例如早餐吃薄煎饼的过程 制作松饼的难点在于知道先做哪一步.从图7-18可知,可以首先加热平底锅或者混合原材料.我们借助拓扑排序这种图算法来确定制作松饼的步 ...

  8. Uva10305 Ordering Tasks

    John有n个任务,但是有些任务需要在做完另外一些任务后才能做. 输入 输入有多组数据,每组数据第一行有两个整数1 <= n <= 100 和 m.n是任务个数(标记为1到n),m两个任务 ...

  9. 拓扑排序(三)之 Java详解

    前面分别介绍了拓扑排序的C和C++实现,本文通过Java实现拓扑排序. 目录 1. 拓扑排序介绍 2. 拓扑排序的算法图解 3. 拓扑排序的代码说明 4. 拓扑排序的完整源码和测试程序 转载请注明出处 ...

随机推荐

  1. 【bzoj2783】[JLOI2012]树 树上倍增

    题目描述 在这个问题中,给定一个值S和一棵树.在树的每个节点有一个正整数,问有多少条路径的节点总和达到S.路径中节点的深度必须是升序的.节点1是根节点,根的深度是0,它的儿子节点的深度为1.路径不必一 ...

  2. Android中有哪些好的开发框架?

    在安卓开发中,框架的使用必不可少,合理利用一些好的开发框架,往往可以达到事半功倍的效果.本文小编就将和大家分享安卓开发者不得不知的5款框架,一起来看看吧,新技能get走起~~ 1.thinkAndro ...

  3. cdq分治入门学习 cogs 1752 Mokia nwerc 2015-2016 G 二维偏序

    /* CDQ分治的对象是时间. 即对于一个时间段[L, R],我们取mid = (L + R) / 2. 分治的每层只考虑mid之前的修改对mid之后的查询的贡献,然后递归到[L,mid],(mid, ...

  4. 根据约束id名找到表名

    数据库运行删除某条数据出现 [Err] ORA-02292: 违反完整约束条件 (ITOUCH_FDA.FKC7DB45E29C5A81ED) - 已找到子集 根据ITOUCH_FDA.FKC7DB4 ...

  5. 插头dp题表

    bzoj1814: Ural 1519 Formula 1 bzoj3125: CITY bzoj1210: [HNOI2004]邮递员 bzoj2331: [SCOI2011]地板 bzoj1187 ...

  6. 【BZOJ 4514】[Sdoi2016]数字配对 费用流

    利用spfa流的性质,我直接拆两半,正解分奇偶(妙),而且判断是否整除且质数我用的是暴力根号,整洁判断质数个数差一(其他非spfa流怎么做?) #include <cstdio> #inc ...

  7. HDU 1394 Minimum Inversion Number(树状数组/归并排序实现

    Minimum Inversion Number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java ...

  8. 使用UMeditor富文本编辑器上传图片

    注:本文系作者原创,但可随意转载. 最近写自己的网站玩儿,写到博客的部分,打算使用UMeditor,因为之前也用过(但是好像没实现图片上传的功能),感觉用起来还比较简单. 不过还是折腾了一下午...遇 ...

  9. php 计算两个日期的间隔天数

    使用php内部自带函数实现 1.使用DateTime::diff 实现计算 参考阅读>>PHP DateTime::diff() 上代码: <?php $start = " ...

  10. Idea 部署非Maven项目

    参考:http://m.blog.csdn.net/z69183787/article/details/78030857 以前一直很好奇,在idea中运行tomcat,把项目部署到其中,运行起来,然后 ...