Luogu P2149 [SDOI2009]Elaxia的路线 | 图论
题解:
题面中给了最简洁清晰的题目描述:“求无向图中,两对点间最短路的最长公共路径”。
对于这个问题我们可以先考虑图中的哪些边对这两对点的最短路产生了贡献。
比如说下面这个图:

我们要求从1到8的最短路和从3到5的最短路的最长公共路径。
先考虑有哪些边对从1到8的最短路产生了贡献,并按这些边被经过时的方向构造一个有向图。
如下:

然后再考虑有哪些边对从3到5的最短路产生了贡献,然后也按这些边被经过时的方向构造一个有向图。
如下:

我们知道,这两对点间最短路的最长公共路径一定在上述两张有向图的重叠部分中。
于是我们可以试着将那两张有向图重叠起来:

接着提取其中的公共部分:

由于我们一开始构造的两张图都一定是有向无环图(想想为什么),所以它们的公共部分也一定是若干张有向无环图。
又因为两点间的最短路一定是一条链,所以两条最短路的公共部分也一定是一条链。
这样的话,我们的任务就变成了在这若干张有向无环图中求最长链。
这个问题可以用dp在O(n)的时间内解决。
那么如何我们该如何构造一开始的两张有向图呢?这可以用SPFA解决。
对于每个点,都记录它是从哪个点走过来的,最后从终点倒推回去即可。
值得注意的是,我们不能只对其中一对点的一个点构造有向图。因为在最短路上走的方向可能是不同的,如果只构造一张图的话就会漏掉行走方向不同的边。
像上面我们举的那个例子,这条边就没有被算进去,但它其实也是这两对点间最短路的公共部分。

那么我们能不能直接忽略图的方向呢?
不行,因为如果直接忽略图的方向的话,我们得出来的答案可能会属于一对点之间的两条不同的最短路(想想为什么)。
对此,我们可以对其中一对点的最短路正着构造出一个有向图,反着再构造出一个有向图,然后将它们分别与对另一对点的最短路构造的有向图进行相应的操作即可。
比如说上面所举的那个例子,我们先对结点1到结点8的最短路构造一个有向图(称为图1),然后再分别对结点3到结点5的最短路和结点5到结点3的最短路构造一个有向图(分别称为图2和图3)。最后,我们先拿图1和图2做一次那样的操作,然后再拿图1和图3做一次那样的操作即可。(“那样的操作”是指提取公共部分和dp求最长链。)
总时间复杂度为O(nm+n2+n),其中的nm是SPFA在最坏情况下的时间复杂度,n2是提取两张有向图公共部分的时间复杂度,n是dp求最长链的时间复杂度。
这个时间复杂度虽然理论上来说是不能过的,但在实际测试中,SPFA的时间复杂度是低于O(nm)的上限的,所以这是能过的。
大家也可以用使用堆优化的dijkstra算法来代替SPFA,以换取在求最短路时O(nlog2n)的稳定时间复杂度。
提取两张有向图公共部分的时间其实也是可以优化的,但这个实现起来较为简单,就不再赘述了。
代码:
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int n=0,m=0,x1=0,y1=0,x2=0,y2=0,ans=0,head=0,tail=0;
//last[i][j]=1表示编号为i的结点可以从编号为j的结点走过来
int d[1505],k[1505],dis[1505],book[1505],que[2265025],last[1505][1505],f[1505][1505],s[1505][1505],h[1505][1505],p[1505][1505];
void SPFA(int s)//SPFA
{
head=1,tail=0;
for(int i=1;i<=n;i++) dis[i]=1e8,book[i]=k[i]=0;
dis[s]=0;
que[++tail]=s;
book[s]=1;
while(head<=tail)
{
int u=que[head++];
book[u]=false;
for(int i=1;i<=n;i++)
{
if(i==u) continue;
if(dis[u]+f[u][i]<dis[i])
{
dis[i]=dis[u]+f[u][i];
k[i]=1;
last[i][k[i]]=u;//更新last数组
if(book[i]==false)
{
que[++tail]=i;
book[i]=true;
}
}
else if(dis[u]+f[u][i]==dis[i]) last[i][++k[i]]=u;//更新last数组
}
}
}
void work(int x,int h[][1505])//构造对最短路有贡献的边所组成的有向图
{
if(book[x]==1) return;
book[x]=1;
for(int i=1;i<=k[x];i++)
{
h[last[x][i]][x]=1;
work(last[x][i],h);
}
}
void dp(int x)//dp求有向无环图最长链
{
if(d[x]>0) return;
for(int i=1;i<=n;i++)
if(p[x][i]==1)
{
dp(i);
d[x]=max(d[x],d[i]+f[x][i]);
}
}
int main()
{
scanf("%d%d%d%d%d%d",&n,&m,&x1,&y1,&x2,&y2);
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
f[i][j]=(i==j)?0:1e8;
for(int i=1;i<=m;i++)
{
int u=0,v=0,w=0;
scanf("%d%d%d",&u,&v,&w);
f[u][v]=f[v][u]=min(f[u][v],w);//邻接矩阵存图
}
//对从x1到y1的最短路进行处理
SPFA(x1);
memset(s,0,sizeof(s));
memset(book,0,sizeof(book));
work(y1,s);
//对从x2到y2的最短路进行处理
SPFA(x2);
memset(h,0,sizeof(h));
memset(book,0,sizeof(book));
work(y2,h);
//提取两张有向图的公共部分
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
if(s[i][j]==1&&h[i][j]==1) p[i][j]=1;
else p[i][j]=0;
//dp求最长链
memset(d,0,sizeof(d));
for(int i=1;i<=n;i++)
if(d[i]==0) dp(i);
for(int i=1;i<=n;i++) ans=max(ans,d[i]);
//对从y2到x2的最短路进行处理
SPFA(y2);
memset(h,0,sizeof(h));
memset(book,0,sizeof(book));
work(x2,h);
//提取两张有向图公共部分
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
if(s[i][j]==1&&h[i][j]==1) p[i][j]=1;
else p[i][j]=0;
//dp求最长链
memset(d,0,sizeof(d));
for(int i=1;i<=n;i++)
if(d[i]==0) dp(i);
for(int i=1;i<=n;i++) ans=max(ans,d[i]);
//输出
printf("%d",ans);
return 0;
}
Luogu P2149 [SDOI2009]Elaxia的路线 | 图论的更多相关文章
- Luogu P2149 [SDOI2009]Elaxia的路线(最短路+记忆化搜索)
P2149 [SDOI2009]Elaxia的路线 题意 题目描述 最近,\(Elaxia\)和\(w**\)的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们必须合理地安排两个人在一起的 ...
- 洛谷 P2149 [SDOI2009]Elaxia的路线 解题报告
P2149 [SDOI2009]Elaxia的路线 题目描述 最近,Elaxia和w**的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间. Elaxia ...
- 洛谷——P2149 [SDOI2009]Elaxia的路线
P2149 [SDOI2009]Elaxia的路线 题目描述 最近,Elaxia和w的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间.Elaxia和w每 ...
- 洛谷—— P2149 [SDOI2009]Elaxia的路线
https://www.luogu.org/problem/show?pid=2149 题目描述 最近,Elaxia和w的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两 ...
- P2149 [SDOI2009]Elaxia的路线
题目描述 最近,Elaxia和w**的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间. Elaxia和w**每天都要奔波于宿舍和实验室之间,他们 希望在 ...
- 洛谷 P2149 [SDOI2009]Elaxia的路线
题目描述 最近,Elaxia和w的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间.Elaxia和w每天都要奔波于宿舍和实验室之间,他们 希望在节约时间的 ...
- P2149 [SDOI2009]Elaxia的路线[最长公共路径]
题目描述 最近,Elaxia和w**的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间. Elaxia和w**每天都要奔波于宿舍和实验室之间,他们 希望在 ...
- Luogu 2149 [SDOI2009]Elaxia的路线
感觉这题可以模板化. 听说spfa死了,所以要练堆优化dijkstra. 首先对$x_{1},y_{1},x_{2},y_{2}$各跑一遍最短路,然后扫一遍所有边看看是不是同时在两个点的最短路里面,如 ...
- BZOJ 1880: [Sdoi2009]Elaxia的路线( 最短路 + dp )
找出同时在他们最短路上的边(dijkstra + dfs), 组成新图, 新图DAG的最长路就是答案...因为两人走同一条路但是不同方向也可以, 所以要把一种一个的s,t换一下再更新一次答案 ---- ...
随机推荐
- HDFS基本命令
1.创建目录 hadoop dfs -mkdir /data hadoop dfs -mkdir -p /data/data1 创建多级目录 2.查看文件 hadoop dfs -ls / 3.上 ...
- liunx常见指令
linux目录结构 bin:存储普通命令 sbin:存储超级命令 home:存储普通用户 root:存储超级用户 usr /usr/local:下存储数据或软件,通常软件都放在其中 tmp:临时目录 ...
- Jmeter系类(32) - JSR223(2) | Groovy常见内置函数及调用
常见内置函数及调用 获取相关函数 获取返回数据并转换为String字符串 prev.getResponseDataAsString() 例子 String Responsedata = prev.ge ...
- Shell系列(9)- 用户自定义变量(2)
定义变量 变量名=变量值 例如: x=123 mulu="当前目录下有 $(ls)" 备注: 变量名只能是字母.下划线.数字组成且不能以数字开头 变量等号两侧不能加空格 若变量值中 ...
- 有哪些浏览器还支持flash?
Flash是大名鼎鼎的全家桶公司Adobe设计的一款网页动画软件,早期的动态网页基本都是基于Flash开发的.但是后来不断爆出关于Flash的安全漏洞和运行效率问题,虽然Adobe公司一直在尝试解决, ...
- localStorage util
// localStorage util var db ={ set : function(key, obj){ localStorage.setItem(key, JSON.stringify(ob ...
- JMeter脚本开发
什么是jmeter脚本 用户操作系统的动作流程 用户操作系统的请求 类似演戏的剧本 怎么快速开发漂亮的jmeter脚本 准确 快速 漂亮,脚本逻辑清晰,维护性高 脚本开发方案 代理 http代理服务器 ...
- 『GoLang』string及其相关操作
目录 1. 字符串简介 2. 字符串的拼接 3. 有关 string 的常用处理 3.1 strings 包 3.1.1 判断两个 utf-8 编码字符串是否相同 3.1.2 判断字符串 str 是否 ...
- 鸿蒙内核源码分析(特殊进程篇) | 龙生龙,凤生凤,老鼠生儿会打洞 | 百篇博客分析OpenHarmony源码 | v46.02
百篇博客系列篇.本篇为: v46.xx 鸿蒙内核源码分析(特殊进程篇) | 龙生龙凤生凤老鼠生儿会打洞 | 51.c.h .o 进程管理相关篇为: v02.xx 鸿蒙内核源码分析(进程管理篇) | 谁 ...
- 前端规范之JS代码规范(ESLint + Prettier)
代码规范是软件开发领域经久不衰的话题,几乎所有工程师在开发过程中都会遇到或思考过这一问题.而随着前端应用的大型化和复杂化,越来越多的前端团队也开始重视代码规范.同样,前段时间,笔者所在的团队也开展了一 ...