【u019】排序(sort)
【问题描述】
一个不同的值的升序排序数列指的是一个从左到右元素依次增大的序列,例如,一个有序的数列A,B,C,D 表示A<B,B<C,C<D。在这道题中,我们将给你一系列形如A<B的关系,并要求你判断是否能够根据这些关系确定这个数列的顺序。
【输入文件】
第一行有两个整数n,m,n表示需要排序的元素数量,2<=n<=26,第1到n个元素将用大写的A,B,C,D....表示。m表示将给出的形如A<B的关系的数量。
接下来有m行,每行有3个字符,分别为一个大写字母,一个<符号,一个大写字母,表示两个元素之间的关系。
【输出文件】
若根据前x个关系即可确定这n个元素的顺序yyy..y(如ABC),输出
Sorted sequence determined after xxx relations: yyy...y.
若根据前x个关系即发现存在矛盾(如A<B,B<C,C<A),输出
Inconsistency found after 2 relations.
若根据这m个关系无法确定这n个元素的顺序,输出
Sorted sequence cannot be determined.
(提示:确定n个元素的顺序后即可结束程序,可以不用考虑确定顺序之后出现矛盾的情况)
【样例输入】
1:
4 6
A<B
A<C
B<C
C<D
B<D
A<B
2:
3 2
A<B
B<A
3:
26 1
A<Z
【样例输出】
1:
Sorted sequence determined after 4 relations: ABCD.
2:
Inconsistency found after 2 relations.
3:
Sorted sequence cannot be determined.
【题解】
这题要用到拓扑排序。
首先把输入的关系看做是一条条边。
A..Z对应数字1..26;
然后所有点的入度一开始为-1.表示都没有出现过。
然后A<B,则为一条从A指向B的有向边。
然后B的入度递增。
然后A的入度如果为-1则改为0,表示其出现过。
然后若是加入这条边之后有某个点没有出现过。那就不可能构成序列。
然后是判环的方法。
加入一个变量num.记录的是那些加入过topsort队列中的元素(用队列来完成topsort)
在进行完topsort之后。
记录入度信息的数组不要着急还原成原来的样子。
然后如果num<n(说明没有完成topsort),且在1..n中发现有1个节点它的入度是大于0的,那就说明出现了环。(如果是因为有元素没有出现而不能完成topsort则所有存在的点在尝试topsort后它们的入度都会等于0)
不能单纯地写成if (不存在未出现的点 && num < n) 来判断是否有环。
因为就算存在未出现的点也可能有环。比如n=5.
你只要在前4个点之间弄一个环。所得到的num依然是小于5.且存在未出现的点。
然后如果进行topsort的时候。发现尾节点和头结点的差在递增头结点之后还是大于等于1.则不能够确定该序列。因为这就说明同时出现了两个入度为0的点。而这是不允许的!
因为这样你就不能确定下一个数字是啥了。
然后会出现重边的情况。
然后输出完整序列的时候要加上末尾的点号!!!
【代码】
#include <cstdio>
#include <cstring> int n, m, ru[27],dl[10000],head,tail,w[27][27]; int main()
{
memset(ru, 255, sizeof(ru));//一开始赋值为-1是表示没有出现过某个点。
scanf("%d%d", &n, &m);
for (int i = 1; i <= m; i++)
{
char s[20];
scanf("%s", s);
int x = s[0] - 'A' + 1, y = s[2] - 'A' + 1;
if (!w[x][y])//会有重边!!!
{
if (ru[y] == -1) //如果y的入度为-1则改为0表示出现过。
ru[y] = 0;
if (ru[x] == -1)//同理x如果出现了也改为0
ru[x] = 0;
w[x][y] = 1;//一条有向边由x指向y
ru[y]++;//y的入度递增、
}
int zero = 0,fushu = 0;//统计入度为0和没有出现的点的个数。
bool notenough = false;//判断能不能得出序列。
head = 0, tail = 0;
for (int j = 1; j <= n; j++)
if (ru[j] == 0)//入度为0则递增zero
{
zero++;
tail++;//同时把该点加入队列中进行topsort
dl[tail] = j;
}
else
if (ru[j] == -1)//找到没有出现的点也记录。
fushu++;
if (zero == 0)//如果没有一个入度为0的点。则说明出现了环。则输出错误信息即可。
{
printf("Inconsistency found after %d relations.", i);
return 0;
}
if (zero > 1 || fushu >0)//如果入度为0的点大于1个或有未出现的点。则说明不足以确定序列。
notenough = true;//但是还要判断是不是出现了环。
int num = tail;//num是进行topsort的点的个数。
int tempru[27];
for (int i = 1; i <= n; i++)//因为接下来的topsort会改变ru数组。所以先用temp记录一下原来的ru
tempru[i] = ru[i];
while (head != tail)
{
head++;
if (tail > head) //如果递增了头结点 尾节点还大于头结点。则说明有2个入度为0的点同时出现。
notenough = true;//则不足以构成序列。
int x = dl[head];
ru[x] = -1;//把这个点的入度置为-1
for (int j = 1; j <= n;j++)//然后找其出度
if (w[x][j] == 1)
{
ru[j]--;//出度的入度递减
if (ru[j] == 0)//如果也变成0了。则加入到队尾
{
num++;//进行过topsort的点数递增;
tail++;
dl[tail] = j;
}
}
}
if (num < n)//不能进行完整的topsort
{
for (int j = 1; j <= n; j++)
if (ru[j] > 0)
{
printf("Inconsistency found after %d relations.", i);
return 0;
}
}
for (int i = 1; i <= n; i++)//把ru之前的状态回溯一下。
ru[i] = tempru[i];
if (notenough && i == m)//如果不足以判断且已经是最后一条边了。则输出无法判断
printf("Sorted sequence cannot be determined.");
if (num == n && !notenough)//如果所有的点都进行了topsort。且足以判断。则输出信息。
{
printf("Sorted sequence determined after %d relations: ", i);
for (int j = 1; j <= tail; j++)//把topsort序列输出一遍。
{
char key = dl[j] + 'A' - 1;
putchar(key);
}
putchar('.');//这个点号也是很坑啊。。。
return 0;
}
}
return 0;
}
【u019】排序(sort)的更多相关文章
- Lucene 排序 Sort与SortField
在sql语句中,有升序和降序排列.在Lucene中,同样也有. Sort里的属性 SortField里的属性 含义 Sort.INDEXORDER SortField.FIELD_DOC 按照索引的顺 ...
- 转:详细解说 STL 排序(Sort)
详细解说 STL 排序(Sort) 详细解说 STL 排序(Sort) 作者Winter 详细解说 STL 排序(Sort) 0 前言: STL,为什么你必须掌握 1 STL提供的Sort 算法 1. ...
- 设计模式 - 模板方法模式(template method pattern) 排序(sort) 具体解释
模板方法模式(template method pattern) 排序(sort) 具体解释 本文地址: http://blog.csdn.net/caroline_wendy 參考模板方法模式(tem ...
- [js] - 关于js的排序sort
js的排序sort并不能一次排序好 function solution(nums){ return nums.sort(sortNumber); } function sortNumber(a, b) ...
- 给乱序的链表排序 · Sort List, 链表重排reorder list LoLn...
链表排序 · Sort List [抄题]: [思维问题]: [一句话思路]: [输入量]:空: 正常情况:特大:特小:程序里处理到的特殊情况:异常情况(不合法不合理的输入): [画图]: quick ...
- 详细解说 STL 排序(Sort)(转)
作者Winter 详细解说 STL 排序(Sort) 0 前言: STL,为什么你必须掌握 1 STL提供的Sort 算法 1.1 所有sort算法介绍 1.2 sort 中的比较函数 1.3 sor ...
- Excel VBA解读(54):排序——Sort方法
Excel VBA解读(54):排序——Sort方法 看看下面的Excel界面截图,“排序”和“筛选”往往在一起,这大概是很多数据需要先排序后筛选吧 首先以“性别”作为排序字段,升序排列,并且第一行 ...
- sort排序,按指定字段进去重,sort -t "^" -k 8 -su,ls给文件名中数字排序sort -k1.5n,Tab符要转义
sort sort 命令对 File 参数指定的文件中的行排序,并将结果写到标准输出.如果 File 参数指定多个文件,那么 sort 命令将这些文件连接起来,并当作一个文件进行排序. sort语法 ...
- 详细解说 STL 排序(Sort)
0 前言: STL,为什么你必须掌握 对于程序员来说,数据结构是必修的一门课.从查找到排序,从链表到二叉树,几乎所有的算法和原理都需要理解,理解不了也要死记硬背下来.幸运的是这些理论都已经比较成熟,算 ...
随机推荐
- BZOJ3277: 串(后缀自动机,Parent树,Dfs序)
Description 字符串是oi界常考的问题.现在给定你n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中 至少k个字符串的子串(注意包括本身). Input 第一行两个整数n, ...
- CISP/CISA 每日一题 14
CISA 每日一题(答) 自动无人值守运行(LIGHTS-OUT)优势:1.信息系统运行成本的遏制/减少:2.持续运行(24/7):3.减少系统错误和中断次数. I/O 控制人员负责保证:1.批处理信 ...
- 【hdu 6181】Two Paths
[链接]http://acm.hdu.edu.cn/showproblem.php?pid=6181 [题意] 让你求从1到n的次短路 [题解] 模板题; 因为点可以重复走; 则一定会有次短路. di ...
- 设计模式六大原则(三):依赖倒置原则(Dependence Inversion Principle)
依赖倒置原则(DIP)定义: 高层模块不应该依赖低层模块,二者都应该依赖其抽象:抽象不应该依赖细节:细节应该依赖抽象. 问题由来: 类A直接依赖类B,假如要将类A改为依赖类C,则必须通过修改类A的代码 ...
- 洛谷 P1207 [USACO1.2]双重回文数 Dual Palindromes
P1207 [USACO1.2]双重回文数 Dual Palindromes 题目描述 如果一个数从左往右读和从右往左读都是一样,那么这个数就叫做“回文数”.例如,12321就是一个回文数,而7777 ...
- POJ--1753--Flip Game【DFS】
链接:http://poj.org/problem? id=1753 题意:一个4*4的方格,有白棋或者黑棋.每次操作是一个位置的颜色翻转,即白变黑.黑变白,而且与它相邻的四个位置的颜色也都跟着改变, ...
- Accelerated C++:通过演示样例进行编程实践——练习解答(第9章)
我的Github地址:https://github.com/lanbeilyj/Accerlerated-C-plus-plus 9-0. Compile, execute, and test the ...
- 早该知道的 7 个JavaScript 技巧[转]
简洁写法 对象的简写在过去,如果你想创建一个对象,你需要这样: var car = new Object(); car.colour = 'red'; car.wheels = 4; car.h ...
- Android前后端交互细节--Json转化为对象的原理
移动互联网用户基数越来越大,除了一些工具类(指南针.手电筒等)的应用,绝大部分APP都需要与后端进行交互. 交互的数据格式有JSON.XML等,由于JSON具有语法简单.占用空间小等优势,基本所有的公 ...
- 深度解析VC中的消息
消息是指什么? 消息系统对于一个win32程序来说十分重要,它是一个程序运行的动力源泉.一个消息,是系统定义的一个32位的值,他唯一的定义了一个事件,向Windows发出一个通知,告诉应用程序某个事情 ...