POJ1094 拓扑排序

问题:POJ1094

本题考查拓扑排序算法

拓扑排序:

1)找到入度为0的点,加入已排序列表末尾;

2)删除该点,更新入度数组。

循环1)2)直到

1. 所有点都被删除,则找到一个拓扑排序;

2. 或剩余结点中没有入度为0的点,则原图中必存在环。

本题算法

1.依次输入一组关系

对当前关系进行拓扑排序

1)若存在环,则无法排序

2)若根据当前关系,每次循环都唯一的确定入度为0的点,则存在排序

2.若输完所有的大小关系之后,仍然没有确定大小排序,也没有发现回环,则排序无法唯一确定

AC代码:

 1 //Memory: 224K        Time: 0MS
 2 #include <iostream>
 3 #include <cstdio>
 4 #include <cstring>
 5 #include <string>
 6
 7 using namespace std;
 8
 9 const int maxn = 26;
10 bool g[maxn][maxn];
11 int indegree[maxn];
12 int in[maxn];
13 int vis[maxn];
14 int flag;
15 int n, m;
16 int out[maxn];
17 string s;
18 int size;
19 int ans;
20
21 int topologicalSort()
22 {
23     memset(vis, 0, sizeof(vis));
24     memcpy(in, indegree, sizeof(in));
25     size = 0;
26     flag = 0;
27     while (size < n){
28         int num = 0;
29         int next;
30         for (int i = 0; i < n; i++){
31             if (vis[i]) continue;
32             if (in[i] == 0) {
33                 num++;
34                 if (num > 1) break;
35                 next = i;
36             }
37         }
38         if (num == 0) return 0;
39         else {
40             out[size++] = next;
41             vis[next] = 1;
42             for (int i = 0; i < n; i++) {
43                 if ( g[next][i] && !vis[i] )
44                     in[i]--;
45             }
46             if (num > 1)
47                 flag = 1;
48         }
49     }
50     if (flag == 0) return 1;
51     else return 2;
52 }
53
54 int main()
55 {
56     while (cin >> n >> m && n) {
57         memset(g, 0, sizeof(g));
58         memset(indegree, 0, sizeof(indegree));
59         ans = 2;
60         for (int i = 0; i < m; i++){
61             cin >> s;
62             if ( ans != 2 ) continue;
63             if (!g[s[0] - ‘A‘][s[2] - ‘A‘]) {
64                 g[s[0] - ‘A‘][s[2] - ‘A‘] = 1;
65                 indegree[s[2] - ‘A‘]++;
66                 ans = topologicalSort();
67
68                 if (ans == 0) {
69                     cout << "Inconsistency found after " << i + 1 <<" relations." << endl;
70                 }
71                 else if ( ans == 1 ){
72                     cout << "Sorted sequence determined after " << i + 1 << " relations: ";
73                     for (int i = 0; i < n; i++)
74                         cout << (char)(out[i] + ‘A‘);
75                     cout << "." << endl;
76                 }
77             }
78         }
79
80         if ( ans == 2 )
81             cout << "Sorted sequence cannot be determined." << endl;
82     }
83     return 0;
84 }

POJ1094 拓扑排序,布布扣,bubuko.com

时间: 2024-11-05 02:35:32

POJ1094 拓扑排序的相关文章

POJ1094拓扑排序

每次输入的时候 进行一次 拓扑排序. 拓扑排序时,每次查找入度为零的点只有一个,则此解为确定解.若找到确定解之后出现环(坑),继续输入,不管它. #include <cstdio> #include <cstring> #include <algorithm> #include <climits> #include <string> #include <iostream> #include <map> #include &

nyoj349 poj1094 Sorting It All Out(拓扑排序)

nyoj349   http://acm.nyist.net/JudgeOnline/problem.php?pid=349poj1094   http://poj.org/problem?id=1094这两个题是一样的,不过在poj上A了才算真的过,ny上数据有一点弱. 题目大意输入n,m. 一共有n个字母(从A开始), m行语句每个语句“x﹤y”,说明x,y之间的偏序关系.让你判断是否可以通过这些关系得到一个唯一的升序序列,若能则输出这个序列并指出通过前多少条语句得出的,如果n个字母间存在矛

拓扑排序讲解

在这里我们要说的拓扑排序是有前提的 我们在这里说的拓扑排序是基于有向无环图的!!!. (⊙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 每学期末,都是万众瞩目的选课时间,由于人数过多,某学校的服务器常常被无数的学生挤的爆掉,这是,教务系统大人说,你们选个课都这么慢,居然还怪我们.于是,每次教务系统都会在服务器快要瘫痪前关闭它.在无数学生的强烈抗议下,教务系统妥协了,再给每个人一次机会,但他让我们用最快的方式决定该选的课程,选上后就退出. 这让大一学渣狗犯了难,在新的选