POJ 1094 (传递闭包 + 拓扑排序)
题目大意:有 1 ~ N 个大写字母,且从 A 开始依次 N 个。再给你 M 个小于的关系,比如 A < B ,让你判断三种可能:
1、在第 i 个关系罗列之后,是否可以满足使得这 N 个字母能递增关系。
2、在第 i 个罗列之后,是否会出现矛盾,例如 A > B,而在第 i 个状态出现后,B > A ,故矛盾。
3、如果 M 个条件罗列完后都没有出现矛盾,且还无法判断 N 个字母的排列顺序,则输出 Sorted sequence cannot be determined.
在前两种情况中,输出最先满足的 i ,也就是说,按 m 个状态的顺序,满足任意一个条件后,其他条件都不用再判断。
思路与分析:
对于 A < B,我们建一个 A --> B 的有向图。
按 M 个状态的顺序,每次得到 A < B ,标记 a[A][B] 为 true,表示 A 能到达 B ,然后全图跑一遍 floyd 传递闭包,判断在第 i 个状态时,是否满足前两种情况。
1、在三层循环传递闭包结束后,判断图中任意两点间是否存在 A > B 且 B < A 的这种矛盾关系,即判断全图两点是否会有 a[i][j] = true 且 a[j][i] = true ,有的话,则判断为第二种情况,标记或输出当前 i 。
2、还需要判断的是,如果 a[i][j]==0 且 a[j][i]==0 ,则说明此时 i 与 j 点之间没有任何小于或大于关系,故在当前状态时,还未能判断出 N 个字母的关系。
可以先用数组存 M 个状态,或者是边输入边判断。但一定要注意的是,如果 floyd 判断为 false (即上一段中的两种情况),则还需要再判断任意两点 i j ,是否为上文中的第一种情况(即 a[i][j]==a[j][i]==true),是的话,则说明为题目所描述的第二种情况。
边输入边判断:
#include<iostream>
#include<algorithm>
#include<string.h>
#include<queue>
#define inf 0x3f3f3f3f
#define maxn 100
using namespace std;
int n,m,cnt,b,w,res;
int head[maxn],in[maxn];
bool a[maxn][maxn];
char c[maxn];
struct Edge
{
int to;
int next;
}edge[maxn*maxn*];
inline void add(int u,int v)
{
edge[++cnt].to=v;
edge[cnt].next=head[u];
head[u]=cnt;
return;
}
inline bool floyd(int C)
{
for(int k=;k<=n;k++){
for(int i=;i<=n;i++){
for(int j=;j<=n;j++){
if(a[i][k]&&a[k][j]) a[i][j]=true;
}
}
}
for(int i=;i<=n;i++){
for(int j=;j<=n;j++){
if(i==j) continue;
if((a[i][j]&&a[j][i])){
b=C;
return false;
}
if(a[i][j]==&&a[j][i]==) return false;
}
}
return true;
}
void solve()
{
queue<int> q;
while(!q.empty()) q.pop();
for(int i=;i<=n;i++) {if(!in[i]) q.push(i);}
int tot=;
while(!q.empty())
{
int x=q.front();
q.pop();
c[tot++]=(char)(x+'A'-);
for(int i=head[x];i;i=edge[i].next){
int v=edge[i].to;
in[v]--;
if(!in[v]) q.push(v);
}
}
c[tot]='\0';
return;
}
void init()
{
b=cnt=w=res=;
memset(c,,sizeof(c));
for(int i=;i<=n;i++) {
head[i]=in[i]=;
for(int j=;j<=n;j++){
a[i][j]=false;
}
}
}
int main()
{
//freopen("test.in","r",stdin);
while(~scanf("%d%d",&n,&m)){
if(n==&&m==) break;
init();
char s[];
for(int i=;i<=m;i++){
scanf("%s",s);
if(w||b>=inf) continue;
int A=s[]-'A'+,B=s[]-'A'+;
a[A][B]=true;
add(A,B),in[B]++;
if(floyd(inf)){
w=;
if(!b) b=i;
continue;
}
else{//可以不需要再进行一遍判断,只需要 floyd 保存 b 之后,最后返回即可。因为可能会先被 a[i][j]==a[j][i]==0 先返回而 b 未被赋值为 inf
for(int k=;k<=n;k++){
for(int j=;j<=n;j++){
if(k==j) continue;
if(a[k][j]&&a[j][k]) {res=i;b=inf;}
}
}
}
}
if(w){
solve();
printf("Sorted sequence determined after %d relations: %s.\n", b,c);
}
else{
if(b){printf("Inconsistency found after %d relations.\n", res);}
else printf("Sorted sequence cannot be determined.\n");
}
}
}
存数组再遍历 M 个状态:
#include<iostream>
#include<algorithm>
#include<string.h>
#include<queue>
#define inf 0x3f3f3f3f
#define maxn 30
using namespace std;
int n,m,cnt,b,w;
int head[maxn],in[maxn];
bool a[maxn][maxn];
char c[maxn],s[][];
struct Edge
{
int to;
int next;
}edge[maxn*maxn*];
inline void add(int u,int v)
{
edge[++cnt].to=v;
edge[cnt].next=head[u];
head[u]=cnt;
return;
}
inline bool floyd(int C)
{
for(int k=;k<=n;k++){
for(int i=;i<=n;i++){
for(int j=;j<=n;j++){
if(a[i][k]&&a[k][j]) a[i][j]=true;
}
}
}
for(int i=;i<=n;i++){
for(int j=;j<=n;j++){
if(i==j) continue;
if((a[i][j]&&a[j][i])||(a[i][j]==&&a[j][i]==)) return false;
}
}
return true;
}
void solve()
{
queue<int> q;
while(!q.empty()) q.pop();
for(int i=;i<=n;i++) {if(!in[i]) q.push(i);}
int tot=;
while(!q.empty())
{
int x=q.front();
q.pop();
c[tot++]=(char)(x+'A'-);
for(int i=head[x];i;i=edge[i].next){
int v=edge[i].to;
in[v]--;
if(!in[v]) q.push(v);
}
}
c[tot]='\0';
return;
}
void init()
{
b=cnt=w=;
memset(c,,sizeof(c));
memset(s,,sizeof(s));
memset(a,,sizeof(a));
memset(head,,sizeof(head));
}
int main()
{
//freopen("test.in","r",stdin);
while(~scanf("%d%d",&n,&m)){
if(n==&&m==) break;
init();
for(int i=;i<=m;i++){
scanf("%s",s[i]);
}
for(int i=;i<=m;i++){
int A=s[i][]-'A'+,B=s[i][]-'A'+;
a[A][B]=true;
if(floyd()){
for(int j=;j<=i;j++){
A=s[j][]-'A'+,B=s[j][]-'A'+;
add(A,B),in[B]++;
}
solve();
printf("Sorted sequence determined after %d relations: %s.\n",i,c);
w=;
}
else{
for(int k=;k<=n;k++){
for(int j=;j<=n;j++){
if(k==j) continue;
if(a[k][j]&&a[j][k]){
printf("Inconsistency found after %d relations.\n", i);
w=;
break;
}
if(w) break;
}
}
}
if(w) break;
}
if(!w) printf("Sorted sequence cannot be determined.\n");
}
}
POJ 1094 (传递闭包 + 拓扑排序)的更多相关文章
- poj 1094(拓扑排序)
http://poj.org/problem?id=1094 题意:给你m个字母,有n个判断语句.求在哪个语句就可以判断出这个是不是一个环,或者在哪个语句可以判断出这些字母的排序规则,或者就是不能确定 ...
- 【POJ 1094】拓扑排序
题意 给出n,代表有以A开始的n个字母,给出它们的m个小于关系(A<B).如果前i个关系可以确定n个字母的一个顺序就输出: Sorted sequence determined after i ...
- Poj(2367),拓扑排序
题目链接:http://poj.org/problem?id=2367 题意: 知道一个数n, 然后n行,编号1到n, 每行输入几个数,该行的编号排在这几个数前面,输出一种符合要求的编号名次排序. 拓 ...
- poj 1270(dfs+拓扑排序)
题目链接:http://poj.org/problem?id=1270 思路:就是一简单的dfs+拓扑排序,然后就是按字典序输出所有的情况. http://paste.ubuntu.com/59872 ...
- poj 3683(2-sat+拓扑排序)
Priest John's Busiest Day Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 11127 Accep ...
- ACM: poj 1094 Sorting It All Out - 拓扑排序
poj 1094 Sorting It All Out Time Limit:1000MS Memory Limit:10000KB 64bit IO Format:%lld & ...
- poj 1094 Sorting It All Out (拓扑排序)
http://poj.org/problem?id=1094 Sorting It All Out Time Limit: 1000MS Memory Limit: 10000K Total Su ...
- [ACM_模拟] POJ 1094 Sorting It All Out (拓扑排序+Floyd算法 判断关系是否矛盾或统一)
Description An ascending sorted sequence of distinct values is one in which some form of a less-than ...
- 【POJ】1094 Sorting It All Out(拓扑排序)
http://poj.org/problem?id=1094 原来拓扑序可以这样做,原来一直sb的用白书上说的dfs............ 拓扑序只要每次将入度为0的点加入栈,然后每次拓展维护入度即 ...
随机推荐
- Numpy数值类型与数值运算-03
什么是NumPy? NumPy是Python中科学计算的基本软件包.它是一个Python库,提供多维数组对象,各种派生对象(例如蒙版数组和矩阵) 以及各种例程,用于对数组进行快速操作,包括数学,逻辑, ...
- Exploratory Testing 3.0 - 探索式测试
最近看了James Bach新发的一篇文章,名为Exploratory Testing 3.0,文章链接:http://www.satisfice.com/blog/archives/1509 这篇文 ...
- javascript split() 把一个字符串分割成字符串数组,类似于PHP的 explode()函数
用法类似于框里的 例子:
- oracle学习笔记(十七) PL/SQL高级应用
PL/SQL高级应用 动态SQL 在PL/SQL中,不能直接执行DDL(create,alter,drop),得使用动态SQL,当然,除了DDL,动态SQL也可以执行DML(select,insert ...
- python处理oracle数据库的返回数据
上代码: import SqlHelper.ORACLE as ORA import pandas as pd if __name__ == '__main__': #连接数据库 ms = ORA.O ...
- Docker配置yapi接口
一.安装Docker 第一步:yum 包更新到最新 sudo yum update 第二步:安装需要的软件包 yum-util 提供yum-config-manager功能,另外两个是devicema ...
- 字体图标引入到HTML---复制用代码
第一步:在样式里面声明字体: 告诉别人我们自己定义的字体 ps: font-family的名字可以自己起 src是引入了4种字体,不用修改的 @font-face { font-family: 'ic ...
- 网站的favicon图标
网站的favicon图标 favicon.ico一般用于作为缩略的网站标志,它显示在浏览器的地址栏或者标签上. 制作favicon图标 把图片转换为png图片 把png图片转换为ico图标,这需要借助 ...
- 在 VS Code 中遇到的一些问题
1.在安装时未配置右键快捷菜单,想重新添加 最简单的就是重新安装一遍,在安装过程中选择好. 其次可以通过以下注册表脚本导入(保存为 .reg 文件),注意因为有中文字符,需要使用记事本保存为 ANSI ...
- 六个office免费学习的精品教程 office自学教程让你从小白到高手
今天给大家分享六个免费学习的精品教程,包括了word,ppt,excel常用的办公三剑客,学会这三个让你办公无忧,也能让你靠这些技术找到一份好工作,这些office自学教程让你从小白到高手. 一.PP ...