Floyd判联通(传递闭包)

Floyd传递闭包顾名思义就是把判最短路的代码替换成了判是否连通的代码,它可以用来判断图中两点是否连通。板子大概是这个样的:

for(int k=1; k<=n; k++){
for(int i=1; i<=n; i++){
for(int j=1; j<=n; j++){
// 把数值计算替换成逻辑运算——就一行,非常简便
e[i][j] = e[i][j] || (e[i][k] && e[k][j]);
}
}
}

题目描述

给定 n个变量和 m个不等式。其中 n小于等于 26,变量分别用前 n的大写英文字母表示。

不等式之间具有传递性,即若 A>B 且 B>C,则 A>C。

请从前往后遍历每对关系,每次遍历时判断:

  • 如果能够确定全部关系且无矛盾,则结束循环,输出确定的次序;
  • 如果发生矛盾,则结束循环,输出有矛盾;
  • 如果循环结束时没有发生上述两种情况,则输出无定解。

    输入格式

    输入包含多组测试数据。

每组测试数据,第一行包含两个整数 n和 m。

接下来 m行,每行包含一个不等式,不等式全部为小于关系。

当输入一行 0 0 时,表示输入终止。

输出格式

每组数据输出一个占一行的结果。

结果可能为下列三种之一:

  1. 如果可以确定两两之间的关系,则输出 "Sorted sequence determined after t relations: yyy...y.",其中't'指迭代次数,'yyy...y'是指升序排列的所有变量。
  2. 如果有矛盾,则输出: "Inconsistency found after t relations.",其中't'指迭代次数。
  3. 如果没有矛盾,且不能确定两两之间的关系,则输出 "Sorted sequence cannot be determined."。

那么我们可以分析题目:题目说要“从前往后遍历每对关系” 那么就不是一次性导入所有数据了,而是每输入一个就计算一遍。

graph TB
Begin(开始) --> A[输入不等式]
A -->E{是否存在矛盾}
E -->|存在|B[输出矛盾信息]
E -->|不存在|C
C{是否能确定两两关系}
C -->|能确定|D[输出升序排列]
C -->|不能确定|A
F{全部不等式输入完成且未发生以上情况} -->G[输出]
A -->F

那么怎么判断是否存在矛盾呢?想想看,不就是既有\(A>B\) 又有\(B>A\)吗。那么就可以在floyd的同时加入判断。

for(int k=1; k<=n; k++){
for(int i=1; i<=n; i++){
for(int j=1; j<=n; j++){
e[i][j] = e[i][j] || (e[i][k] && e[k][j]);
// 注意要i!=j
// 如果e[i][j]和e[j][i]都联通肯定存在矛盾
if(e[i][j] && e[j][i] && i!=j){
data = 0;
}
}
}
}

那怎么判断能否确定两两关系呢?那就是在没有矛盾的前提下,两两首尾相连。如果存在两个点没有首尾相连的情况,那肯定不行的。我这里把判断它的代码单独拿了出来放在一个函数里,因为如果在floyd中写的话它是会变化的,可能在某次循环时它不连通,但循环几次后它又联通了。所以还不如拿出去.

bool check(){
for(int i=1; i<=n; i++){
for(int j=1; j<=n; j++){
if(!e[i][j] && !e[j][i] && i!=j) return 0;
}
}
return 1;
}

可以判断两两关系了,那怎么打印出次序呢?我在某个大佬那里受到启发——观察矩阵。试想一下,如果\(A<B<C<D\),那么A联通的点有3个,B联通的点有2个,C联通的点有…… 也就是这个样子的:

A[1][n] 0 1 1 1
B[2][n] 0 0 1 1
C[3][n] 0 0 0 1
D[4][n] 0 0 0 0

那么就只用依次取出“1”最多的打印出来就好。

inline void out(){
// #define p pair<int, char>
priority_queue<p, vector<p>, less<p> > q;
int t;
for(int i=1; i<=n; i++){
t = 0;
for(int j=1; j<=n; j++){
if(i != j) t += e[i][j];
}
q.push( (p){t, i-1+'A'} );
}
while(!q.empty()){
printf("%c",q.top().second);
q.pop();
}
printf(".\n");
}

这道题个人感觉非常nice,他开拓了我们的新思路:观察矩阵(找规律)。好了,以上是我的全部理解。博客freshman,如有错误,还请指点!

AC代码:仅供参考

点击查看代码
#include<bits/stdc++.h>
using namespace std;
#define p pair<int, char>
int n,m,data;
bool e[27][27],node[27];
string s;
inline void floyd(){
for(int k=1; k<=n; k++){
for(int i=1; i<=n; i++){
for(int j=1; j<=n; j++){
e[i][j] = e[i][j] || (e[i][k] && e[k][j]);
if(e[i][j] && e[j][i] && i!=j){
data = 0;
}
}
}
}
}
bool check(){
for(int i=1; i<=n; i++){
for(int j=1; j<=n; j++){
if(!e[i][j] && !e[j][i] && i!=j) return 0;
}
}
return 1;
}
inline void out(){
priority_queue<p, vector<p>, less<p> > q;
int t;
for(int i=1; i<=n; i++){
t = 0;
for(int j=1; j<=n; j++){
if(i != j) t += e[i][j];
}
q.push( (p){t, i-1+'A'} );
}
while(!q.empty()){
printf("%c",q.top().second);
q.pop();
}
printf(".\n");
}
int main(){
while(scanf("%d%d",&n,&m) && n){
memset(e, 0, sizeof(e));
memset(node, 0, sizeof(node));
data = 1;
for(int i=1; i<=m; i++){
cin>>s;
e[s[0]-'A'+1][s[2]-'A'+1] = 1;
node[s[0]-'A'+1] = node[s[2]-'A'+1] = 1;
if(data == 1){
floyd();
if(data == 0){
printf("Inconsistency found after %d relations.\n",i);
//break;
}
else if(check()){
printf("Sorted sequence determined after %d relations: ",i);
out();
data = 2;
//break;
}
}
}
if(data == 1){
printf("Sorted sequence cannot be determined.\n");
}
}
}

Floyd判联通(传递闭包) & poj1049 sorting it all out的更多相关文章

  1. UVA - 247 Calling Circles Floyd判圈

    思路:利用的Floyd判圈,如果i能到j,j也能到i说明i和j在同一个圈里.每个人的名字可用map编号.最后DFS打印答案即可. AC代码 #include <cstdio> #inclu ...

  2. UVa11549计算器谜题[floyd判圈]

    题意: 有个老式计算器,每次只能记住一个数字的前n位.现在输入一个整数k,然后反复平方,一直做下去,能得到的最大数是多少.例如,n=1,k=6,那么一次显示:6,3,9,1... 白书上的题 set, ...

  3. SGU 455 Sequence analysis(Cycle detection,floyd判圈算法)

    题目链接:http://acm.sgu.ru/problem.php?contest=0&problem=455 Due to the slow 'mod' and 'div' operati ...

  4. UVa 1594 (Floyd判圈) Ducci Sequence

    大白书上P42那个计算器的题目就用到了这个办法,Floyd判圈法. 当然,用STL里的map也是可以的. #include <cstdio> #include <cmath> ...

  5. UVA 11549 CALCULATOR CONUNDRUM(Floyd判圈算法)

    CALCULATOR CONUNDRUM   Alice got a hold of an old calculator that can display n digits. She was bore ...

  6. uva11549 Floyd判圈法

    题意: 给两个数n, k,每次将k平方取k的前n位,问所有出现过的数的最大值 原来这就是floyd判圈法.. #include<cstdio> #include<cstdlib> ...

  7. UVA 11549 Calculator Conundrum (Floyd判圈算法)

    题意:有个老式计算器,每次只能记住一个数字的前n位.现在输入一个整数k,然后反复平方,一直做下去,能得到的最大数是多少.例如,n=1,k=6,那么一次显示:6,3,9,1... 思路:这个题一定会出现 ...

  8. leetcode202(Floyd判圈算法(龟兔赛跑算法))

    Write an algorithm to determine if a number is "happy". 写出一个算法确定一个数是不是快乐数. A happy number ...

  9. Floyd判圈算法

    Floyd判圈算法 leetcode 上 编号为202 的happy number 问题,有点意思.happy number 的定义为: A happy number is a number defi ...

  10. Codeforces Gym 101252D&&floyd判圈算法学习笔记

    一句话题意:x0=1,xi+1=(Axi+xi%B)%C,如果x序列中存在最早的两个相同的元素,输出第二次出现的位置,若在2e7内无解则输出-1. 题解:都不到100天就AFO了才来学这floyd判圈 ...

随机推荐

  1. UniRx-unirx中的对象池

    UniRx-unirx中的对象池 对象池Unirxunity 对象池 一.对象池模式 <游戏设计模式-对象池模式> 1.概念 定义一个池对象,其包含了一组可重用对象. 其中每个可重用对象都 ...

  2. tar命令 --null -T 参数详解

    tar 命令的 --null 和 -T 参数可以一起使用,以从 null 设备读取文件名,并将这些文件名传递给 tar 命令来处理. --null 参数的作用是将文件名作为 null 字符分隔的字符串 ...

  3. ASP.NET Core使用Hangfire定时发布文章

    ASP.NET Core使用Hangfire实现定时任务 前言 也是上了5天班,终于迎来了休息,抽空更新下博客,然后就是下周一公司会对我进行考核,希望考核能通过吧!!! 然后我想给博客添加一个定时发布 ...

  4. Gerrit 大量代码提交流程优化

    # Gerrit 大量代码提交流程优化 背景 本文适用于 提交大量初始代码 的场景(仓库版本的早期). 有时候提交大量代码到Gerrit的时候会提示: $ git push Counting obje ...

  5. TI AM62x工业开发板规格书(单/双/四核ARM Cortex-A53 + 单核ARM Cortex-M4F,主频1.4GHz)

    1 评估板简介 创龙科技TL62x-EVM是一款基于TI Sitara系列AM62x单/双/四核ARM Cortex-A53 + 单核ARM Cortex-M4F多核处理器设计的高性能低功耗工业评估板 ...

  6. FileZilia FATAL ERROR: Network error: Software caused connection abort

    使用FileZilia sftp传文件,对象服务器突然关闭,导致FileZilia传输中断. 等待对象服务器打开后,使用FileZilia想继续传输文件,结果一直显示: FATAL ERROR: Ne ...

  7. 如何让 3D 数字孪生场景闪闪发光

    今日图扑软件功能分享:我们将探讨 HT 系统如何通过分组管理灯光.裁切体和流光,以提高场景光影效果的精准度和整体可控性. HT 中的灯光.裁切体.流光是会影响它所在区域一定范围内的其他节点的表现,如 ...

  8. SQLServer的操作以及一些概念

    初始SQLServer 创建数据库 create database 数据库名 创建表 选择数据库 USE MyDb; 创建表 CREATE TABLE t_user ( id INT, usernam ...

  9. 1. C++ 开发环境

    C++ 开发环境 Visual C++ / GCC(G++) / Clang(Clang++) 集成开发环境:Visual Studio / CodeLite / Code::blocks / CLi ...

  10. 转载 | win11右键菜单改为win10的bat命令(以及恢复方法bat)

    原文来自这里:https://blog.51cto.com/knifeedge/5340751 版权归:IT利刃出鞘 本质上就是写入注册表. 一.右键菜单改回Win10(展开) 1. 新建文件:win ...