C-3 SRM 08

描述

给一个图,n 个点 m 条双向边,每条边有其长度。n 个点中有 k 个是特殊点,问任意两个特殊点的最短路是多少。

输入格式

第一行三个整数 n m k

第二行 k 个整数 ,为各个特殊点

接下来 m 行,每行三个整数 x y d,表示 x 到 y 有一条长度为 d 的边

输出格式

一个整数

样例输入

5 5 3
1 3 5
1 2 3
2 3 4
3 4 1
4 5 8
1 5 19

样例输出

7

数据范围与约定

  • 图为联通图

样例解释

样例中,1-3 的最短路为 7,3-5 的最短路为 9,1-5 的最短路为 16,因此答案为最小值 7

这道题对我来说有点难啊 想了半天只会跑k-1次最短路 最后在大爷的帮助下还是写出来了 果然自己还是过于蒟蒻QAQ

这道题呢 我的写法是 将所有的关键点以0作为初始距离 扔进优先队列里面 跑一遍Dijkstra 

记录每个点最近的关键点以及到关键点的距离(dis)

最后枚举边 如果两端的点的最近关键点不一样就更新一波答案

下面证明写法的正确性

我们在跑最短路的时候顺便记录一下每个点是从哪个点扩展来的

首先显然答案不可能比真实的答案偏小 所以证明如果答案存在 则一定能找到

设一个最优解a——b——c——da,d为关键点,

1. dis(a,b)<=dis(a,d)/2

2. dis(c,d)<=dis(a,d)/2

(证明b,c之间有边)

如果因为等距离的问题 b,c最近关键点不是 a,d, 那么只要 b,c 的最近关键点不同,仍可得到答案

如果 b,c 的最近关键点相同 设这个点为x,那么令 x!=a 

那么 dis(a,x)=dis(a,b)+dis(b,x)<=dis(a,b)+dis(b,c)+dis(c,x)=dis(a,b)+dis(b,c)+dis(c,d)

可见 a——b——x 不比 a——b——c——d 差,

如果两个都是最优解 那么由于dis(a,b)==dis(b,x)==dis(a,d)/2

设 a到b路径上最靠近b的点为 e 则dis(e,a)<dus(a,b)=dis(b,x)

所以e的最近关键点不会是x , 那么 (e的最近关键点)——e——b————x这条路同上面两条一样是另一个最优解 且能由e-b更新

证毕

然后就贴一波代码咯 2333

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
const int M=1e5+,inf=1e9+;
int read(){
int ans=,f=,c=getchar();
while(c<''||c>''){if(c=='-') f=-; c=getchar();}
while(c>=''&&c<=''){ans=ans*+(c-''); c=getchar();}
return ans*f;
}
int n,m,k,p[M],dis[M],blong[M];
int cnt,first[M],ans=inf,h[M];
struct note{int to,from,next,w;}e[*M];
void ins(int a,int b,int w){cnt++; e[cnt].to=b; e[cnt].from=a; e[cnt].next=first[a]; e[cnt].w=w; first[a]=cnt;}
void insert(int a,int b,int w){ins(a,b,w); ins(b,a,w);}
struct node{
int d,pos;
bool operator <(const node& x)const{return x.d<d;}
};
priority_queue<node>q;
void dj(){
for(int i=;i<=n;i++) dis[i]=inf;
for(int i=;i<=k;i++){
int now=p[i];
blong[now]=now; dis[now]=;
q.push((node){,now});
}
while(!q.empty()){
node y=q.top(); q.pop();
int x=y.pos;
if(y.d>dis[x]) continue;
for(int i=first[x];i;i=e[i].next){
int now=e[i].to;
if(dis[now]>dis[x]+e[i].w){
dis[now]=dis[x]+e[i].w;
blong[now]=blong[x];
q.push((node){dis[now],now});
}
}
}
}
int main()
{
int x,y,w;
n=read(); m=read(); k=read();
for(int i=;i<=k;i++) p[i]=read(),h[p[i]]=;
for(int i=;i<=m;i++){
x=read(); y=read(); w=read();
insert(x,y,w); if(h[x]&&h[y]) ans=min(ans,w);
}//printf("%d\n",ans);
dj();
for(int i=;i<=cnt;i+=)
if((!h[e[i].to]||!h[e[i].from])&&blong[e[i].from]!=blong[e[i].to])
ans=min(ans,dis[e[i].from]+dis[e[i].to]+e[i].w);
printf("%d\n",ans);
return ;
}

汕头市队赛 SRM 08 C的更多相关文章

  1. 汕头市队赛 SRM 08 B

    B-3 SRM 08 描述 给长度为 n 的数列 A 和长度为 m 的数列 B,问有多少长度为 m 的数列 C 满足 输入格式 第一行俩整数 n 和 m 第二行 n 个整数 ,表示数列 A 第三行 m ...

  2. 汕头市队赛 SRM 08 A

    比赛没参加 所以回来补题咯 A还是自己YY出来了的 可惜比赛没有打 描述 给一个 01 串设为其 S,询问是否存在只出现两次的 01 串 T. 这里的出现定义为存在一串下标 ,满足  且 . 输入格式 ...

  3. 汕头市队赛 SRM 07 D 天才麻将少女kpm

    这道题放了很久还是回来补了 D 天才麻将少女KPM SRM 07 背景&&描述 天才麻将少女KPM立志要在日麻界闯出一番名堂.     KPM上周叒打了n场麻将,但她这次又没控分,而且 ...

  4. 汕头市队赛SRM 20 T2不净的圣杯

    不净的圣杯 SRM 20 背景 作为一张BUG级别的卡,官方打算把它修改得人畜无害一些…… 虽然名字还没想好,但是能力大概是对敌方所有单位造成d点伤害,d为自己牌组中所有卡的编号的最大公约数.这无疑是 ...

  5. 汕头市队赛 SRM 07 B 好玩的麻将

    B 好玩的麻将 SRM 07 背景&&描述 天才麻将少女KPM立志要在日麻界闯出一番名堂.     KPM上周又打了n场麻将,又控了分使得自己的排名是1..n的一个排列.     但她 ...

  6. 汕头市队赛 SRM 06 C 秀恩爱

    C 秀恩爱 SRM 06 背景&&描述         KPM坐在直升机上俯瞰小渔村景象.         渔村可看作二维平面,密密麻麻地到处都是单身狗,KPM当前所在坐标为(sx,s ...

  7. 汕头市队赛 SRM 06 A 撕书

    A 撕书 SRM 06 背景&&描述 游行寺汀正在杀书.         书总共有n页,每页都可以看作是一个小写英文字母,所以我们可以把书看成长度为n的字符串s.         琉璃 ...

  8. 汕头市队赛 SRM 06 B 起伏的排名

    B 起伏的排名 SRM 06 背景&&描述 天才麻将少女KPM立志要在日麻界闯出一番名堂.     在上个星期她打了n场麻将,每场麻将都有n名玩家.KPM自然记得自己的n次排名.   ...

  9. 汕头市队赛 SRM 07 A 你的麻将会排序吗

    A 你的麻将会排序吗 SRM 07 曾经有过一些沉迷日麻的小孩纸,后来呀,他们都去寻找自己的世界了. kpm也是这样的小孩纸.他想有一只自动整理牌的机器.当麻将以给定的顺序进入机器时,通过机器的运转, ...

随机推荐

  1. php 操作 mysql 实现批量执行mysql语句 mysql文件

    <?php /** * 批量运行sql文件 * 正则分隔是重点 preg_split("/;[\r\n]+/", filecontent) */ $config = requ ...

  2. 关于Mysql唯一索引的操作方法(添加删除)

    首先我们查看一下News数据表的索引信息      使用命令 show index from ‘数据表名称’; 目前数据表中仅有一个主键索引 继续,我们给news表添加两个唯一索引(两种方法) 方法一 ...

  3. Pandas基本命令

    关键缩写和包导入 在这个速查手册中,我们使用如下缩写: df:任意的Pandas DataFrame对象 同时我们需要做如下的引入: import pandas as pd 创建测试对象 import ...

  4. POJ 1854 贪心(分治)

    Evil Straw Warts Live Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 1144   Accepted:  ...

  5. Android 数据库中的数据给到ListView

    前言:因为之前学的都是用一个自己定义的类,完成将某一个bean中的数据直接获取,而实际中通常是通过数据库来得到的,总之,最终就是要得到数据.提一下最重要的东西,我把它叫做代理,如同一个校园代理,没有他 ...

  6. 理解Queue队列中join()与task_done()的关系

    在网上大多关于join()与task_done()的结束原话是这样的: Queue.task_done() 在完成一项工作之后,Queue.task_done()函数向任务已经完成的队列发送一个信号 ...

  7. 3. 与服务器对话:理解 HTTP 协议

    0.服务器与本地交换机制 2.详解HTtp服务 (1)与服务器对话的流程 (2)Reque 请求 (3)Response 响应 200 成功 404 没有网页 (4)Get/Post区别 get查询数 ...

  8. 获取ubuntu中软件包的有用地址

    http://us.archive.ubuntu.com/ubuntu/pool/main/g/gettext/

  9. Win10开始菜单中的天气不更新问题的解决方法

    两台电脑同时做的Win10系统,最新的1703 Creator Update 版本,其中一台的开始菜单中天气方块总是显示图标,试了各种方法都不行,最后是点开天气App,在App的顶端有几个按钮,其中有 ...

  10. 《Cracking the Coding Interview》——第13章:C和C++——题目1

    2014-04-25 19:13 题目:用C++写一个读取文件倒数K行的方法. 解法:因为是要取倒数K行,所以我的思路是一行一行地读.过程中需要保存一个长度为K的链表,每次新读到一行都将表头节点移到表 ...