题目链接: 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 (传递闭包 + 拓扑排序)的更多相关文章

  1. poj 1094(拓扑排序)

    http://poj.org/problem?id=1094 题意:给你m个字母,有n个判断语句.求在哪个语句就可以判断出这个是不是一个环,或者在哪个语句可以判断出这些字母的排序规则,或者就是不能确定 ...

  2. 【POJ 1094】拓扑排序

    题意 给出n,代表有以A开始的n个字母,给出它们的m个小于关系(A<B).如果前i个关系可以确定n个字母的一个顺序就输出: Sorted sequence determined after i ...

  3. Poj(2367),拓扑排序

    题目链接:http://poj.org/problem?id=2367 题意: 知道一个数n, 然后n行,编号1到n, 每行输入几个数,该行的编号排在这几个数前面,输出一种符合要求的编号名次排序. 拓 ...

  4. poj 1270(dfs+拓扑排序)

    题目链接:http://poj.org/problem?id=1270 思路:就是一简单的dfs+拓扑排序,然后就是按字典序输出所有的情况. http://paste.ubuntu.com/59872 ...

  5. poj 3683(2-sat+拓扑排序)

    Priest John's Busiest Day Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 11127   Accep ...

  6. ACM: poj 1094 Sorting It All Out - 拓扑排序

    poj 1094 Sorting It All Out Time Limit:1000MS     Memory Limit:10000KB     64bit IO Format:%lld & ...

  7. poj 1094 Sorting It All Out (拓扑排序)

    http://poj.org/problem?id=1094 Sorting It All Out Time Limit: 1000MS   Memory Limit: 10000K Total Su ...

  8. [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 ...

  9. 【POJ】1094 Sorting It All Out(拓扑排序)

    http://poj.org/problem?id=1094 原来拓扑序可以这样做,原来一直sb的用白书上说的dfs............ 拓扑序只要每次将入度为0的点加入栈,然后每次拓展维护入度即 ...

随机推荐

  1. 拎壶学python3-----(5)pycharm解决运行时少库的问题

    有时候我们运行的时候会出现如下现象 no module named requests怎么办呢? 点击设置: 安装成功后会显示如下: 怎么进行安装呢? 然后退出再运行: 发现已经没有缺少requests ...

  2. PHP面试题2019年奇虎360面试题及答案解析

    一.单选题(共29题,每题5分) 1.以下代码 a.php 输出的结果是? a.php 的代码如下: b.php的代码如下: A.foo in a B.什么也不输出 C.报错 D.foo in b 参 ...

  3. 用AI思维给成本降温,腾讯WeTest兼容性测试直击底价!

    WeTest 导读 当AI成为各行业提高产业效率的动能,很多人开始疑惑,这架智能化的“无人机”何时在移动应用测试中真正落地?在今年的国际数码互动娱乐博览会(ChinaJoy)上,腾讯WeTest给出了 ...

  4. Android native进程间通信实例-binder篇之——解决实际问题inputreader内建类清楚缓存

    我在实际开发中,遇到一个问题,在电容屏驱动中没有发送input_sync 给上层,导致电容屏有的数据缓存在inputreader 中,会导致系统一系列奇怪问题发生, 至于为什么驱动不发送input_s ...

  5. phpstorm xdebug 无法断点调试问题

    最近用phpstorm+xdebug调试一段php代码的时候莫名其妙的无法切入断点调试 本地用的php集成环境是UPUPW ANK V1.1.7 64Bit 选择了集成环境中php版本为7.1.x 搞 ...

  6. Node.js实现热加载

    不管是node.js原生开发,还是借助express,kora等框架开发node.js的情况下,在对代码做出更新后,都是需要重启已生效我们的文件的. 本文记录一次在原生node.js开发的时候,为项目 ...

  7. Linux PXE + Kickstart 自动装机

    大规模装机时,使用无人值守装机便可大大简便人工操作,提高效率. PXE 网络安装 配置双网卡 这里ens33为nat网络,ens37为仅主机网络,配置ens37 [root@localhost ~]# ...

  8. Pycharm 2019 添加 docker 解释器

    打开docker的tls

  9. 4. jenkins 构建任务

    这边说一下 ,服务器的分布.  因为我这边 测试环境的项目比较多,不能修改主机名 1号机器 节点机器(node) 最终项目发布到这台机器 2号机器  jenkins jenkins的部署,发布 3号机 ...

  10. win10安装配置mongodb

    1. 下载MongoDB并安装官网下载地址:https://blog.csdn.net/qq_41127332/article/details/80755595 ,选择合适的版本进行下载.我选择是3. ...