拓扑排序欢乐多

拓扑排序

一:使用DFS实现

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. #define maxn 10000 + 10
  4. int c[maxn],topo[maxn], k;
  5. int n, r;
  6. vector<int>G[maxn];
  7. int dfs(int u)
  8. {
  9. c[u] = -1;
  10. for(int i=0; i<G[u].size(); i++)
  11. {
  12. int &t = G[u][i];
  13. if(c[t] < 0) return false;
  14. if(!c[t] && !dfs(t)) return false;
  15. }
  16. c[u] = 1;
  17. topo[k--] = u;
  18. return true;
  19. }
  20. bool Topo()
  21. {
  22. for(int i=n; i>=1; i--)
  23. if(!c[i])
  24. {
  25. if(!dfs(i)) return false;
  26. }
  27. return true;
  28. }
  29. int main()
  30. {
  31. while(~scanf("%d%d", &n, &r))
  32. {
  33. int a, b;
  34. k = n;
  35. memset(c, 0, sizeof(c));
  36. for(int i=0; i<r; i++)
  37. {
  38. cin>>a>>b;
  39. G[a].push_back(b);
  40. }
  41. bool flag = Topo();
  42. cout<<flag<<endl;
  43. for(int i=1; i<=n; i++)
  44. cout<<topo[i]<<" ";
  45. }
  46. return 0;
  47. }

Just do it

相关例题

二:使用入度概念以及队列处理

1.使用一般队列

  1. queue<int>q;
  2. //priority_queue<int,vector<int>,greater<int>>q;
  3. //优先队列的话,会按照数值大小有顺序的输出
  4. //此处为了理解,暂时就用简单队列
  5. int topo()
  6. {
  7. for(int i=1; i<=n; i++)
  8. {
  9. if(indegree[i]==0)
  10. {
  11. q.push(i);
  12. }
  13. }
  14. int temp;
  15. while(!q.empty())
  16. {
  17. temp=q.front();//如果是优先队列,这里可以是top()
  18. printf("%d->",temp);
  19. q.pop();
  20. for(inti=1; i<=n; i++) //遍历从temp出发的每一条边,入度--
  21. {
  22. if(map[temp][i])
  23. {
  24. indegree[i]--;
  25. if(indegree[i]==0)q.push(i);
  26. }
  27. }
  28. }
  29. }

2.使用优先队列(这里定义越大的整数拥有越大的优先级)

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. #define maxn 100000+10
  4. struct cmp
  5. {
  6. bool operator()(int a, int b)
  7. {
  8. return a < b;
  9. }
  10. };
  11. priority_queue<int, vector<int>, cmp>Q;
  12. int deg[maxn];
  13. vector<int>G[maxn];
  14. int main()
  15. {
  16. int n, r;
  17. int u, v;
  18. cin>>n>>r;
  19. for(int i=0; i<r; i++)
  20. {
  21. cin>>u>>v;
  22. G[u].push_back(v);
  23. deg[v]++;
  24. }
  25. for(int i=1; i<=n; i++)
  26. if(deg[i] == 0)
  27. {
  28. Q.push(i);
  29. }
  30. while(!Q.empty())
  31. {
  32. int t = Q.top();///因为使用优先队列 必须在与其相关节点入队之前将其pop这样不会因为优先性对于拓扑序有所影响
  33. ///换而言之,当入度为0时该节点已经是自由的,所有在这之前的节点已经得到安排,即拓扑序已经得到实现
  34. cout<<t<<" ";
  35. Q.pop();
  36. for(int i=0; i<G[t].size(); i++)
  37. {
  38. int &tmp = G[t][i];
  39. deg[tmp]--;
  40. if(!deg[tmp]) Q.push(tmp);
  41. }
  42. }
  43. return 0;
  44. }
时间: 2024-12-07 08:40:22

拓扑排序欢乐多的相关文章

拓扑排序讲解

在这里我们要说的拓扑排序是有前提的 我们在这里说的拓扑排序是基于有向无环图的!!!. (⊙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.尽管还有可能其他的可能,你只需找出其中的一个即可. 输入: 输入为一