P3905 道路重建

题目描述

从前,在一个王国中,在\(n\)个城市间有\(m\)条道路连接,而且任意两个城市之间至多有一条道路直接相连。在经过一次严重的战争之后,有\(d\)条道路被破坏了。国王想要修复国家的道路系统,现在有两个重要城市\(A\)和\(B\)之间的交通中断,国王希望尽快的恢复两个城市之间的连接。你的任务就是修复一些道路使\(A\)与\(B\)之间的连接恢复,并要求修复的道路长度最小。

输入格式

输入文件第一行为一个整数\(n(2<n≤100)\),表示城市的个数。这些城市编号从\(1\)到\(n\)。

第二行为一个整数\(m(n-1≤m≤\frac{1}{2} n(n-1))\),表示道路的数目。

接下来的\(m\)行,每行\(3\)个整数\(i,j,k(1≤i,j≤n,i≠j,0<k≤100)\),表示城市\(i\)与\(j\)之间有一条长为\(k\)的道路相连。

接下来一行为一个整数\(d(1≤d≤m)\),表示战后被破坏的道路的数目。在接下来的\(d\)行中,每行两个整数\(i\)和\(j\),表示城市\(i\)与\(j\)之间直接相连的道路被破坏。

最后一行为两个整数A和B,代表需要恢复交通的两个重要城市。

输出格式

输出文件仅一个整数,表示恢复\(A\)与\(B\)间的交通需要修复的道路总长度的最小值。

输入输出样例

输入 #1

3

2

1 2 1

2 3 2

1

1 2

1 3

输出 #1

1

【三种方法】

1.【最简单的弗洛伊德】

【思路】

弗洛伊德的方法

先将有路的点都连接起来

由于只需要修改损坏的点

所以完好的道路是可以走的而且不需要修复

所以消耗为0

可以把损坏的边标记一下

把没被标记的也就是完好的边改为0

因为不需要消耗

最后再跑一遍弗洛伊德就好了

【完整代码】

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int Max = 102;
int f[Max][Max];
bool use[Max][Max];
int main()
{
int n,m,k;
int x,y,z;
scanf("%d%d",&n,&m);
for(register int i = 1;i <= n;++ i)
for(register int j = 1;j <= n;++ j)
f[i][j] = 999;
for(register int i = 1;i <= m;++ i)
{
cin >> x >> y >> z;
f[x][y] = f[y][x] = z;
}
scanf("%d",&k);
for(register int i = 1;i <= k;++ i)
{
cin >> x >> y;
use[x][y] = use[y][x] = true;
}
for(register int i = 1;i <= n;++ i)
for(register int j = 1;j <= n;++ j)
if(use[i][j] == false && f[i][j] != 999)
f[i][j] = 0;
for(register int k = 1;k <= n;++ k)
for(register int i = 1;i <= n;++ i)
for(register int j = 1;j <= n;++ j)
f[j][i] = f[i][j] = min(f[i][j],f[i][k] + f[k][j]);
int A,B;
scanf("%d%d",&A,&B);
cout << f[A][B] << endl;
return 0;
}

2.【SPFA】

【思路】

SPFA

SPFA诈尸+1

先输入有路的数据

建立边但是先不赋值

保持他默认为0的距离

只把某条边对应的长度稍微记录一下

然后输入坏掉的路

这个时候才将坏掉的路的长度赋值上去

因为完好无损的路可以通过而且不需要耗费去修复

所以对需要修复的路径的总长度没有贡献

就是0

但是坏掉的不一样

贡献它本身的长度

然后跑SPFA求出A到B的最短路就好了

【完整代码】

#include<iostream>
#include<cstdio>
#include<queue>
using namespace std;
const int Max = 102;
struct node
{
int y;
int ne;
int z;
}a[Max * Max];
int sum = 0;
int head[Max];
int acioi[Max * Max];
void add(int x,int y,int z)
{
a[++ sum].y = y;
a[sum].ne = head[x];
acioi[sum] = z;
head[x] = sum;
} int d[Max];
bool use[Max];
int A,B;
int n,m,k;
void SPFA()
{
queue<int>q;
q.push(A);
for(register int i = 1;i <= n;++ i)
d[i] = 999;
d[A] = 0;
while(!q.empty())
{
int qwq = q.front();
q.pop();use[qwq] = false;
for(register int i = head[qwq];i != 0;i = a[i].ne)
{
int awa = a[i].y;
if(d[awa] > d[qwq] + a[i].z)
{
d[awa] = d[qwq] + a[i].z;
if(use[awa] == false)
{
use[awa] = true;
q.push(awa);
}
}
}
}
}
int main()
{
scanf("%d%d",&n,&m);
int x,y,z;
for(register int i = 1;i <= m;++ i)
{
scanf("%d%d%d",&x,&y,&z);
add(x,y,z);add(y,x,z);
}
scanf("%d",&k);
for(register int i = 1;i <= k;++ i)
{
scanf("%d%d",&x,&y);
for(register int j = head[x];j != 0;j = a[j].ne)
if(a[j].y == y)
a[j].z = acioi[j];
for(register int j = head[y];j != 0;j = a[j].ne)
if(a[j].y == x)
a[j].z = acioi[j];
}
scanf("%d%d",&A,&B);
SPFA();
cout << d[B] << endl;
return 0;
}

3.【dijstra】

【思路】

地杰斯特拉+堆优化

我知道的三个求最短路的方法里面貌似生存能力最强的一个

输入数据

处理方式和SPFA的方法一个样

先只把边连接起来但是不赋值边权

让边权保持为0

然后将损坏掉的路赋值上边权

(原因不多赘述了,前面两种方法都说过了)

然后跑dijkstra就好了

【完整代码】

#include<iostream>
#include<cstdio>
#include<queue>
using namespace std;
struct point
{
int w;//按照w从小到大排序
int x;
bool operator < (const point & xx)const
{
return xx.w < w;
}
};
priority_queue<point>q;
const int Max = 102;
struct node
{
int y,ne,z;
}a[Max * Max];
int sum = 0;
int head[Max];
int acioi[Max * Max];
int n,m;
int A,B;
int dis[Max];
bool use[Max]; void add(int x,int y,int z)
{
a[++ sum].y = y;
a[sum].ne = head[x];
acioi[sum] = z;
head[x] = sum;
} void dj()
{
dis[A] = 0;
q.push((point){0,A});
while(!q.empty())
{
point qwq = q.top();
q.pop();
int x = qwq.x;
if(use[x] == true)
continue;
else
use[x] = true;
for(register int i = head[x];i != 0;i = a[i].ne)
{
int awa = a[i].y;
if(dis[awa] > dis[x] + a[i].z)
{
dis[awa] = dis[x] + a[i].z;
if(use[awa] == false)
q.push((point){dis[awa],awa});
}
}
}
} int main()
{
scanf("%d%d",&n,&m);
for(register int i = 1;i <= n;++ i)
dis[i] = 999;
int x,y,z;
for(register int i = 1;i <= m;++ i)
{
scanf("%d%d%d",&x,&y,&z);
add(x,y,z);add(y,x,z);
}
int k;
scanf("%d",&k);
for(register int i = 1;i <= k;++ i)
{
scanf("%d%d",&x,&y);
for(register int j = head[x];j != 0;j = a[j].ne)
if(a[j].y == y)
a[j].z = acioi[j];
for(register int j = head[y];j != 0;j = a[j].ne)
if(a[j].y == x)
a[j].z = acioi[j];
}
scanf("%d%d",&A,&B);
dj();
cout << dis[B] << endl;
return 0;
}

洛谷 P3905 道路重建 题解的更多相关文章

  1. 洛谷——P3905 道路重建

    P3905 道路重建 题目描述 从前,在一个王国中,在n个城市间有m条道路连接,而且任意两个城市之间至多有一条道路直接相连.在经过一次严重的战争之后,有d条道路被破坏了.国王想要修复国家的道路系统,现 ...

  2. 洛谷 P3905 道路重建

    题目描述 从前,在一个王国中,在n个城市间有m条道路连接,而且任意两个城市之间至多有一条道路直接相连.在经过一次严重的战争之后,有d条道路被破坏了.国王想要修复国家的道路系统,现在有两个重要城市A和B ...

  3. 洛谷P3905 道路重建

    题目:https://www.luogu.org/problemnew/show/P3905 分析: 此题是显然的最短路算法,只是看到一起删掉的一堆边感到十分棘手,而且还要求出的是最短添加边的总长度 ...

  4. 洛谷 P4198 楼房重建 题解

    题面 首先你要知道题问的是什么:使用一种数据结构,动态地维护以1为起点地最长上升子序列(把楼房的高度转化成斜率地序列)的长度: 怎么做?线段树! 我们在线段树上维护两个东西:1.这个区间内斜率的最大值 ...

  5. P1359 租用游艇 && P3905 道路重建 ------Floyd算法

    P1359 租用游艇   原题链接https://www.luogu.org/problemnew/show/P1359 P3905 道路重建   原题链接https://www.luogu.org/ ...

  6. 洛谷P1119-灾后重建-floyd算法

    洛谷P1119-灾后重建 题目描述 给出\(B\)地区的村庄数NN,村庄编号从\(0\)到\(N-1\),和所有\(M\)条公路的长度,公路是双向的. 给出第\(i\)个村庄重建完成的时间\(t_i\ ...

  7. 洛谷P2832 行路难 分析+题解代码【玄学最短路】

    洛谷P2832 行路难 分析+题解代码[玄学最短路] 题目背景: 小X来到了山区,领略山林之乐.在他乐以忘忧之时,他突然发现,开学迫在眉睫 题目描述: 山区有n座山.山之间有m条羊肠小道,每条连接两座 ...

  8. 【洛谷P3960】列队题解

    [洛谷P3960]列队题解 题目链接 题意: Sylvia 是一个热爱学习的女孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Sylvia 所在的方阵中有 n×m ...

  9. P3905 道路重建

    P3905 道路重建我一开始想错了,我的是类似kruskal,把毁坏的边从小到大加,并且判断联通性.但是这有一个问题,你可能会多加,就是这条边没用,但是它比较小,你也加上了.居然还有10分,数据也是水 ...

随机推荐

  1. 记CentOS 发布.NET Core 2.0

    centos 7.x sudo rpm --import https://packages.microsoft.com/keys/microsoft.asc sudo sh -c 'echo -e & ...

  2. netcore 实现跨应用的分布式session

    需求场景 网站a,域名为 a.site.com 网站b, 域名为 b.site.com 需要在a.b两个站点之间共享session 解决方案 使用redis作为分布式缓存存储 设置sessionId ...

  3. 玩转dockerfile

    镜像的缓存特性 Docker 会缓存已有镜像的镜像层,构建新镜像时,如果某镜像层已经存在,就直接使用,无需重新创建. 举例说明.在前面的 Dockerfile 中添加一点新内容,往镜像中复制一个文件: ...

  4. 相似文档查找算法之 simHash及其 java 实现

    传统的 hash 算法只负责将原始内容尽量均匀随机地映射为一个签名值,原理上相当于伪随机数产生算法.产生的两个签名,如果相等,说明原始内容在一定概 率 下是相等的:如果不相等,除了说明原始内容不相等外 ...

  5. C++STL—map的使用

    最近写的一道题用到了STL中的map,这部分内容之前没有系统学过,这里就简单归纳一下吧. 资料来源: https://www.w3cschool.cn/cpp/cpp-fu8l2ppt.html ht ...

  6. Python进阶(二)----函数参数,作用域

    Python进阶(二)----函数参数,作用域 一丶形参角度:*args,动态位置传参,**kwargs,动态关键字传参 *args: ​ 动态位置参数. 在函数定义时, * 将实参角度的位置参数聚合 ...

  7. permission

    import 'package:flutter/material.dart'; import 'dart:io'; import 'dart:async'; import 'package:rxdar ...

  8. Java 之 request 案例——用户登录

    用户登录案例需求: 1.编写login.html登录页面 username & password 两个输入框 2.使用Druid数据库连接池技术,操作mysql,db1数据库中user表 3. ...

  9. 如何统一管理单个任务下所有API的同步情况?

    1. 一分钟完成单个API配置 单个API的配置包含:API名称.URL地址.请求方式.参数设置.自定义高级设置. 参数允许用户填写:Text.WebService.Timestamp.DependO ...

  10. Java文件流下载并提示文件不存在

    做文件下载功能的时候,一般使用流的形式下载文件, 如果源文件不存在,下载页面可能就会没有提示,或者一片空白 用户操作之后可能一头雾水,那如何友好提示呢? 想到的有两种 1.可以尝试下载一个名称为:文件 ...