单源最短路问题拓展


Description

给你一张图,图上有 n 个点,m 条边,要你找到两个点,使其最短路恰好包含给定的 k 个点。输出这条最短路的长度,输入保证有解。

输入格式

第一行两个数 n , m。表示有 n 个点,m 条边。

接下来 m 行每行三个数 xi, yi, vi, 表示有一条长度为vi的双向路径连接对应的两个点。

接下来一个数 k。

接下来一行 k 个数,表示一定要包含的点。

输出格式

一个数,符合要求的最短路长度。

样例输入

样例一

6 6

1 2 2

2 6 2

1 3 1

3 4 1

4 5 1

5 6 1

3

5 1 3

样例二

6 6

1 2 2

2 6 2

1 3 1

3 4 1

4 5 1

5 6 1

2

1 6

样例输出

样例一

3

样例二

4

数据范围:

对于30%的数据,1<=N<=10,1<=M<=20

对于60%的数据,1<=N<=500,1<=M<=1000。

对于100%的数据,1<=N<=100000,1<=M<=300000,1<=vi<=10000,1<=K<=N,保证有解。


题解

我们可以很容易地看出,对于满足要求的这条最短路的两个端点一定是属于要包含的点。因为如果不是,我们可以删掉这个端点,得到的解一定会更优。所以,如果我们找到了这两个端点,就可以很容易地求出答案。

那么我们如何找到这两个端点呢?根据题意,所有需要包含的点均在这条路径上,端点一定是其他点能到达的最远的点。所以我们任取一个包含的点,做一次单源最短路,找到离他最远的那个需要包含的点,这一定是一个端点。

我们再以这个端点做一次单源最短路,即可找到另一个端点。同时,另一个端点到此端点的距离已经求出了。所以一共只用做两遍单源最短路即可得出最后的答案。

代码

#include <cstdio>
#include <iostream>
#include <queue>
using namespace std;
#define adde(u,v,w) {e[cnt] = (edge){v,w,head[u]};head[u] = &e[cnt++];} const int maxn = 1e5 + 5, maxm = 3e5 + 5;
const long long inf = 1e12;
int n,m,k;
long long dis[maxn];
int cnt;
int can[maxn];
bool inq[maxn]; queue<int>q; struct edge {
int v;
long long w;
edge *next;
}e[maxm * 2], *head[maxn]; int spfa(int s) {
while(!q.empty())q.pop();
for(int i = 1;i <= n;i++)dis[i] = inf,inq[i] = 0;
dis[s] = 0;
q.push(s);
inq[s] = 1;
while(!q.empty()) {
int u = q.front();q.pop();inq[u] = 0;
for(edge *k = head[u];k;k = k->next) {
if(dis[u] + k->w < dis[k->v]) {
dis[k->v] = dis[u] + k->w;
if(!inq[k->v]) {
inq[k->v] = 1;q.push(k->v);
}
}
}
}
int ans = 0,t = s;
for(int i = 0;i < k;i++)if(dis[can[i]] > ans) ans = dis[can[i]], t = can[i];
return t;
} int main() {
scanf("%d%d",&n,&m);
for(int i = 0;i < m;i++) {
int u,v,w;
scanf("%d%d%d",&u,&v,&w);
adde(u,v,w);adde(v,u,w);
}
scanf("%d",&k);
for(int i = 0;i < k;i++) {
int x;scanf("%d",&x);can[i] = x;
}
int t = spfa(can[0]);
t = spfa(t);
printf("%lld",dis[t]);
return 0;
}

[10.26_P2] 最短路 (单源最短路应用)的更多相关文章

  1. ACM学习历程——HDU5137 How Many Maos Does the Guanxi Worth(14广州10题)(单源最短路)

    Problem Description    "Guanxi" is a very important word in Chinese. It kind of means &quo ...

  2. 用scheme语言实现SPFA算法(单源最短路)

    最近自己陷入了很长时间的学习和思考之中,突然发现好久没有更新博文了,于是便想更新一篇. 这篇文章是我之前程序设计语言课作业中一段代码,用scheme语言实现单源最段路算法.当时的我,花了一整天时间,学 ...

  3. 【UVA1416】(LA4080) Warfare And Logistics (单源最短路)

    题目: Sample Input4 6 10001 3 21 4 42 1 32 3 33 4 14 2 2Sample Output28 38 题意: 给出n个节点m条无向边的图,每条边权都为正.令 ...

  4. 【算法】单源最短路——Dijkstra

    对于固定起点的最短路算法,我们称之为单源最短路算法.单源最短路算法很多,最常见的就是dijkstra算法. dijkstra主要用的是一种贪心的思想,就是说如果i...s...t...j是最短路,那么 ...

  5. HDU-4849 Wow! Such City! (单源最短路)

    Problem Description Doge, tired of being a popular image on internet, is considering moving to anoth ...

  6. 最短路模板(Dijkstra & Dijkstra算法+堆优化 & bellman_ford & 单源最短路SPFA)

    关于几个的区别和联系:http://www.cnblogs.com/zswbky/p/5432353.html d.每组的第一行是三个整数T,S和D,表示有T条路,和草儿家相邻的城市的有S个(草儿家到 ...

  7. [ACM_图论] Domino Effect (POJ1135 Dijkstra算法 SSSP 单源最短路算法 中等 模板)

    Description Did you know that you can use domino bones for other things besides playing Dominoes? Ta ...

  8. 单源最短路_SPFA_C++

    当我们需要求一个点到其它所有点的最短路时,我们可以采用SPFA算法 代码特别好写,而且可以有环,但是不能有负权环,时间复杂度是O(α(n)n),n为边数,α(n)为n的反阿克曼函数,一般小于等于4 模 ...

  9. 【算法系列学习】Dijkstra单源最短路 [kuangbin带你飞]专题四 最短路练习 A - Til the Cows Come Home

    https://vjudge.net/contest/66569#problem/A http://blog.csdn.net/wangjian8006/article/details/7871889 ...

随机推荐

  1. angular2 启动步骤

    以下内容转自网络 1. 创建项目文件夹 创建一个新的文件夹来保存你的项目,比如一开始有个self就好了 2.安装基础库 首先确保已经安装了node.js 我们使用 npm package manage ...

  2. js总结(一):javascript的类型:基本类型、对象和数组

    javascript 类型分为2种,一个是原始值,另一个是复杂值(对象). 一.原始值 5个原始值是:数字,字符,布尔,null,undefined. 9个原生的对象构造函数:Number Strin ...

  3. jQuery 1.9升级指南

    http://www.css88.com/archives/5086 原文地址:http://jquery.com/upgrade-guide/1.9/ 翻译的不对或者不通顺的地方欢迎拍砖留言。 概述 ...

  4. 遇到的Ajax相关问题

  5. NYOJ 237 游戏高手的烦恼

    游戏高手的烦恼时间限制:1000 ms | 内存限制:65535 KB难度:5描述有一位传说级游戏高手,在闲暇时间里玩起了一个小游戏,游戏中,一个n*n的方块形区域里有许多敌人,玩家可以使用炸弹炸掉某 ...

  6. 动手实操:如何用 Python 实现人脸识别,证明这个杨幂是那个杨幂?

    当前,人脸识别应用于许多领域,如支付宝的用户认证,许多的能识别人心情的 AI,也就是人的面部表情,还有能分析人的年龄等等,而这里面有着许多的难度,在这里我想要分享的是一个利用七牛 SDK 简单的实现人 ...

  7. hihoCoder#1127 二分图三·二分图最小点覆盖和最大独立集

    原题地址 主要是介绍了两个定理: 1. 二分图最大匹配数    = 二分图最小点覆盖数 2. 二分图最小点覆盖数 = 二分图顶点数 - 二分图最小点覆盖数 注意,都是二分图 代码:(匈牙利算法) #i ...

  8. 【BZOJ3939】Cow Hopscotch(动态开点线段树)

    题意: 就像人类喜欢跳格子游戏一样,FJ的奶牛们发明了一种新的跳格子游戏.虽然这种接近一吨的笨拙的动物玩跳格子游戏几乎总是不愉快地结束,但是这并没有阻止奶牛们在每天下午参加跳格子游戏  游戏在一个R* ...

  9. oc温习二:基本运算及基本运算符

    C语言一共有34种运算符,如下: 运算符分类: 1.按照功能划分: (1)算术运算符 + 加法运算符 - 减法运算符 或者负值运算符 * 乘法运算符 / 除法运算符 % 模运算符,或者取余运算符,要求 ...

  10. Delphi控件大全

    首先来大体上为控件分一下类,以方便我们后面的讨论.   但因为控件的种类太多,所以就粗略的分为如下几个类别∶   ---界面风格类   ---Shell外观类   ---Editor类   ---Gr ...