浅谈Floyd的三种用法 By cellur925

Floyd大家可能第一时间想到的是他求多源最短路的n³算法。其实它还有另外两种算法的嘛qwq。写一发总结好了qwq。

 

一、多源最短路

放段代码跑,注意枚举顺序,用邻接矩阵存图。本质是一种动规。

复杂度O(n³)。

1 for(int k=1;k<=n;k++)
2     for(int i=1;i<=n;i++)
3         for(int j=1;j<=n;j++)
4             f[i][j]=min(f[i][j],f[i][k]+f[k][j]); 
View Code

放个例题跑。

灾后重建

二、传递闭包

在交际网络中,给定若干个元素,若干个二元关系,关系有传递性。传递闭包就是一种“通过传递性推导出尽量多的元素之间关系的问题”,求出可确定排名的元素个数。

实现用一个布尔型的邻接矩阵,f[i][j]=1表示i与j有关系,否则则没有关系。

我们每次可以枚举k点,来解决那些间接相关的关系处理。

1 for(int k=1;k<=n;k++)
2     for(int i=1;i<=n;i++)
3         for(int j=1;j<=n;j++)
4             f[i][j]|=f[i][k]&f[k][j]; 
View Code

例题 [USACO08JAN]牛大赛Cow Contest

对于奶牛的编程能力,用f[i][j]=1表示i比j强,之后就是一个裸的传递闭包。跑一遍后n²统计每只牛它与其他牛的关系是否已经确定,意思就是说只要有f[i]j]=1或f[j][i]=1其中一个就行,来统计答案。

Code

 1 #include<cstdio>
 2 #include<algorithm>
 3 
 4 using namespace std;
 5 
 6 int n,m,ans;
 7 int f[200][200];
 8 
 9 int main()
10 {
11     scanf("%d%d",&n,&m);
12     for(int i=1;i<=m;i++)
13     {
14         int x=0,y=0;
15         scanf("%d%d",&x,&y);
16         f[x][y]=1;
17     }
18     for(int k=1;k<=n;k++)
19         for(int i=1;i<=n;i++)
20             for(int j=1;j<=n;j++)
21                 f[i][j]|=f[i][k]&f[k][j];
22     for(int i=1;i<=n;i++)
23     {
24         int j;
25         for(j=1;j<=n;j++)
26         {
27             if(i==j) continue;
28             if(f[i][j]==0&&f[j][i]==0) break;
29         }
30         if(j>n) ans++;
31     }
32     printf("%d",ans);
33     return 0;
34 }
View Code

三、求无向图最小环

例题1 USACO4.1篱笆回路 

这道题难在建图,图建好以后就是裸的跑floyd找最小环了。

(瞎说一句,这题竟然有个数组开了1000的空间,但是越界了呀qwq)

Code

 1 /*
 2 ID:cellur_2
 3 TASK:fence6
 4 LANG:C++
 5 */
 6 #include<cstdio>
 7 #include<algorithm>
 8 #include<cstring>
 9 
10 using namespace std;
11 const int inf=0x3f3f3f3f;
12 
13 int n,num,ans=inf;
14 int dis[300][300],mapp[300][300];
15 struct node{
16     int len;
17     int lcnt,rcnt,lid,rid,id;
18     int l[300],r[300];
19 }edge[300];
20 
21 int main()
22 {
23     scanf("%d",&n);
24     for(int i=1;i<=n;i++)
25     {
26         scanf("%d",&edge[i].id);
27         int x=edge[i].id;
28         scanf("%d",&edge[x].len);
29         scanf("%d%d",&edge[x].lcnt,&edge[x].rcnt);
30         for(int j=1;j<=edge[x].lcnt;j++)
31             scanf("%d",&edge[x].l[j]);
32         for(int j=1;j<=edge[x].rcnt;j++)
33             scanf("%d",&edge[x].r[j]);
34     }
35     for(int i=1;i<=n;i++)
36     {// lid  这条边左端点的点编号
37      // rid  这条边右端点的点编号  
38         if(!edge[i].lid) edge[i].lid=++num;
39         for(int j=1;j<=edge[i].lcnt;j++)
40         {
41             int x=edge[i].l[j];
42             bool flag=0;
43             for(int k=1;k<=edge[x].lcnt;k++)
44                 if(edge[x].l[k]==i) 
45                 {
46                     flag=1;
47                     break;
48                 }
49             if(flag) edge[x].lid=edge[i].lid;
50             else edge[x].rid=edge[i].lid;
51         }
52         if(!edge[i].rid) edge[i].rid=++num;
53         for(int j=1;j<=edge[i].rcnt;j++)
54         {
55             int x=edge[i].r[j];
56             bool flag=0;
57             for(int k=1;k<=edge[x].lcnt;k++)
58                 if(edge[x].l[k]==i)
59                 {
60                     flag=1;
61                     break;
62                 }
63             if(flag) edge[x].lid=edge[i].rid;
64             else edge[x].rid=edge[i].rid;
65         }
66     }
67     memset(mapp,0x3f,sizeof(mapp));
68     memset(dis,0x3f,sizeof(dis));
69     ans=dis[2][33];
70     for(int i=1;i<=n;i++) mapp[i][i]=0,dis[i][i]=0;
71     for(int i=1;i<=n;i++)
72     {
73         int lid=edge[i].lid;
74         int rid=edge[i].rid;
75         int len=edge[i].len;
76         mapp[rid][lid]=mapp[lid][rid]=len;
77         dis[rid][lid]=dis[lid][rid]=len;
78     }    
79     //floyd找最小环 
80     for(int k=1;k<=num;k++)
81     {
82         for(int i=1;i<k;i++)
83             for(int j=i+1;j<k;j++)
84                 ans=min(ans,dis[i][j]+mapp[i][k]+mapp[k][j]);
85         for(int i=1;i<=num;i++)
86             for(int j=1;j<=num;j++)
87                 dis[i][j]=min(dis[i][j],dis[i][k]+dis[k][j]);
88     }
89     printf("%d\n",ans);
90     return 0;
91 }
View Code

 例题2 POJ 1734 Sightseeing Trip

其实是floyd找最小环的板子题,但是由于题目要求输出一种合法的方案,所以我们只要再开一个vector就行了。

Code

 1 #include<cstdio>
 2 #include<algorithm>
 3 #include<vector>
 4 #include<cstring>
 5 
 6 using namespace std;
 7 typedef long long ll;
 8 
 9 int n,m;
10 int ans=0x3f3f3f3f;
11 int dis[200][200],mapp[200][200],pos[200][200];
12 vector<int>path;
13 
14 void get_path(int x,int y)
15 {
16     if(pos[x][y]==0) return ;
17     get_path(x,pos[x][y]);
18     path.push_back(pos[x][y]);
19     get_path(pos[x][y],y);
20 }
21 
22 int main()
23 {
24     scanf("%d%d",&n,&m);
25     memset(dis,0x3f,sizeof(dis));
26     for(int i=1;i<=n;i++) dis[i][i]=0;
27     for(int i=1;i<=m;i++)
28     {
29         int x=0,y=0,z=0;
30         scanf("%d%d%d",&x,&y,&z);
31         dis[x][y]=dis[y][x]=min(dis[x][y],z);
32     }
33     memcpy(mapp,dis,sizeof(dis));
34     for(int k=1;k<=n;k++)
35     {
36         for(int i=1;i<k;i++)
37             for(int j=i+1;j<k;j++)
38                 if((ll)mapp[i][j]+dis[j][k]+dis[i][k]<ans) 
39                 {
40                     ans=mapp[i][j]+dis[i][k]+dis[k][j];
41                     path.clear();
42                     path.push_back(i);
43                     get_path(i,j);
44                     path.push_back(j);
45                     path.push_back(k);
46                 }
47         for(int i=1;i<=n;i++)
48             for(int j=1;j<=n;j++)
49                 if(mapp[i][j]>mapp[i][k]+mapp[k][j])
50                 {
51                     mapp[i][j]=mapp[i][k]+mapp[k][j];
52                     pos[i][j]=k;
53                 }
54     }
55     if(ans==0x3f3f3f3f) 
56     {
57         printf("No solution.");
58         return 0;
59     }
60     for(int i=0;i<path.size();i++)
61         printf("%d ",path[i]);
62     return 0;
63 }
View Code

 

转载于:https://www.cnblogs.com/nopartyfoucaodong/p/9590907.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/250653.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

js获取viewbag

后台代码&#xff1a; List<WMSRptJHJY> wpjh WMSRptJHJYManage.GetModelList("");   ViewBag.WPJH Newtonsoft.Json.JsonConvert.SerializeObject(wpjh); 前台代码&#xff1a; var aa Html.Raw(ViewBag.WPJH);   var PWJH eval("(" aa &…

汇编 --- 栈结构的妙用

在js中双重循环,代码如下: for(let i 0; i < l1;i){for(let j0; j< l2;j){// todo...} }以上代码在汇编中是如何实现的呢. 汇编中用cx和loop来代表一次循环 mov cx, 4 s: mov ax,1loop s但是如果使用如下方法实现双重循环将会出错 assume cs:codesg, ds:datasgdatasg …

PHP 安全编程建议

PHP 安全编程建议 简介 要提供互联网服务&#xff0c;当你在开发代码的时候必须时刻保持安全意识。可能大部分 PHP 脚本都对安全问题都不在意&#xff0c;这很大程度上是因为有大量的无经验程序员在使用这门语言。但是&#xff0c;没有理由让你因为对你的代码的不确定性而导致不…

day2-列表、元组、字典、字符串

1.列表&#xff08;list&#xff09; names[悟空,艾玛,克林,龟仙人,天津饭,饺子,乌龟] print(names)---》列表&#xff0c;然后打印。 names[悟空,艾玛,克林,龟仙人,天津饭,饺子,乌龟]# list section section1names[0] #下标从0开始 #print(section1) #结果为悟空seciton2names…

开漏输出、推挽输出的区别

推挽输出:可以输出高,低电平,连接数字器件。 输出 0 时&#xff0c;N-MOS 导通&#xff0c;P-MOS 高阻&#xff0c;输出0。 输出 1 时&#xff0c;N-MOS 高阻&#xff0c;P-MOS 导通&#xff0c;输出1&#xff08;不需要外部上拉电路&#xff09;。 开漏输出:输出端相当于三极管…

罗马数字

古罗马帝国开创了辉煌的人类文明&#xff0c;但他们的数字表示法的确有些繁琐&#xff0c;尤其在表示大数的时候&#xff0c;现在看起来简直不能忍受&#xff0c;所以在现代很少使用了。之所以这样&#xff0c;不是因为发明表示法的人的智力的问题&#xff0c;而是因为一个宗教…

汇编 --- 从磁盘(扇区2到18)上读取数据到内存中

下面代码读取柱面:0,磁头:0,扇区从2到18的数据到内存 0x8200~0xa3ff处 需要明白以下几点: 给定柱面,磁头,一个扇形区域是512字节,对应的物理可以理解为512个灯泡组(一个灯泡组有8个小灯泡)确定读取到内存中的位置 为什么是0x8200:因为0x8000~0x81ff这512个字节要留给启动区.为…

那些值得思考的PHP问题

那些值得思考的PHP问题 1、关于弱类型 函数strpos是返回字符串str2在str1的位置&#xff0c;没有找到则返回false&#xff0c;然而如果在实际应用上返回的位置是0&#xff0c;在if语句中0也被当作false&#xff0c;所以我们需要对false做类型判断&#xff0c; $str1 yabadaba…

Lvs Tun隧道模式配置

######## TUN是IP Tunneling &#xff0c;IP隧道的简称&#xff0c;它将调度器收到的IP数据包封装在一个新的IP数据包中&#xff0c;转交给应用服务器&#xff0c;然后实际服务器的返回数据会直接返回给用户。 工作原理&#xff1a; 用户请求负载均衡服务器&#xff0c;当IP数…

mysql-常用sql

记录下工作中常用的sql 删除重复数据 delete from student where id not in (select min(id) from student group by name); -- 该语句在mysql下会报错&#xff0c; -- 执行报错&#xff1a;1093 - You cant specify target table student for update in FROM clause -- 原因是…

优雅的使用Laravel之phpstorm配置

优雅的使用Laravel之phpstorm配置 先打开一个Laravel 项目&#xff0c;然后在project tool 窗口选择根节点、然后右键->Composer | Init composer 。 如果你的电脑里没有composer.phar&#xff0c;可以点击链接来下载。然后点击项目 composer->add denpendency.. 搜索ba…

算法 --- 求两个集合的并集

const unionL (l1, l2)>{for(let i0; i <l1.length; i){if(l2.indexOf(l1[i]) -1){l2.push(l1[i])}}return l2 }let l1 [1,2,3,4]; let l2 [1,2,5]; console.log(unionL(l1,l2)); //[1, 2, 5, 3, 4]

java实验四——找鞍点

package hello;public class 实验四 {public static void main(String[] args) {// TODO Auto-generated method stubint[][] a {{9,8,6},{2,3,1},{8,5,2}};System.out.println("二维数组为&#xff1a;");for(int i0;i<a.length;i){for(int j0;j<a[i].length;…

BZOJ 4551树题解

好吧&#xff0c;洛谷的数据比较水暴力就可以过。。。。&#xff08;而且跑到飞快&#xff09; 不过&#xff08;BZ水不过去&#xff09;还是讲讲正规的做法。 其实一眼可以看出可以树剖&#xff0c;但是&#xff0c;码起来有点麻烦。 其实有一种更简单的离线做法。 我们很容易…

es6 --- 使用Symbol保护私有变量

定义一个人物类 假设其属性有姓名和性别我们希望,性别在声明后就固定不变 传统方法 var Person (function(){var _gender ;function P(name, gender){this.name name;_gender gender;}P.prototype.getGender function(){return _gender;}return P; })();var p1 new Pe…

组合数

long long factorial(int n) {long long m 1;for(int i1;i<n;i)m*i;return m; } long long C(int n,int m) {return factorial(n)/(factorial(m)*factorial(n-m));//可能会溢出 } 正解&#xff1a; long long C(int n,int m) {if(m<n-m) m n-m;long long ans 1;for(in…

Mysql中的联合索引、前缀索引、覆盖索引

Mysql中的联合索引、前缀索引、覆盖索引 索引 索引是一种特殊的文件&#xff0c;它们包含着对数据表里所有记录的引用指针。更通俗的说&#xff0c;数据库索引好比是一本书前面的目录&#xff0c;能加快数据库的查询速度。 联合索引 又名复合索引&#xff0c;由两个或多个列…

LVM逻辑卷管理

什么是逻辑卷&#xff1f;因为可以将文件系统像卷一样伸长或缩短之故。 LVM做法&#xff0c;将几个物理分区或磁盘&#xff0c;通过软件组合成为一块看起来是独立的大磁盘&#xff08;VG&#xff09;&#xff0c;然后将这块大磁盘再经过分成可使用分区&#xff08;LV&#xff0…

es6 --- 自制迭代器

对象 对象如下 const obj {left: 100,top: 200 }不可迭代 for(let attr of obj){console.log(attr); }迭代规则 可迭代,所具有的属性[Symbol.iterator] 需要自己给obj添加迭代规则 obj[Symbol.iterator] () >{// 获取obj的所有键let keys Object.keys(obj);let len …

软件工程的实践项目课程的自我目标

对实践项目完成后学习到的能力的预期&#xff1a;这算是自己第一次参加的团队合作的软件实践吧&#xff0c;以前自己搞的小“玩意”&#xff0c;实难登大雅之堂&#xff0c;期待实践项目后--->1、自己的代码能力能够有较明显的提高&#xff0c;代码更加规范。 2、提升团队合…