HDU - 3499 Flight 双向SPFA+枚举中间边
Flight
InputThere are no more than 10 test cases. Subsequent test cases are separated by a blank line.
The first line of each test case contains two integers N and M ( 2 <= N <= 100,000
0 <= M <= 500,000 ), representing the number of cities and flights. Each of the following M lines contains "X Y D" representing a flight from city X to city Y with ticket price D ( 1 <= D <= 100,000 ). Notice that not all of the cities will appear in the list! The last line contains "S E" representing the start and end city. X, Y, S, E are all strings consisting of at most 10 alphanumeric characters.
OutputOne line for each test case the least money Shua Shua have to pay. If it's impossible for him to finish the trip, just output -1.Sample Input
- 4 4
- Harbin Beijing 500
- Harbin Shanghai 1000
- Beijing Chengdu 600
- Shanghai Chengdu 400
- Harbin Chengdu
- 4 0
- Harbin Chengdu
Sample Output
- 800
- -1
Hint
- In the first sample, Shua Shua should use the card on the flight from
- Beijing to Chengdu, making the route Harbin->Beijing->Chengdu have the
- least total cost 800. In the second sample, there's no way for him to get to
- Chengdu from Harbin, so -1 is needed.
- 题意:ShuaShua要从一个城市到另一个城市,给出每两城市之间的花费(有向),ShuaShua可以有一次半价的机会,求最小花费。
思路:最短路径问题。开始想到求出最短路后选择其中最大的边/2,但其实错误,很容易举出反例:路径1:1 1 100 路径2:30 30 30 开始时102 90选择花费少的路径2,减半价后52 75路径1花费反而相对更少。
因此我们可以换一种思路,以起点和终点为单源分别求出到各点的最短路,然后枚举每一条边作为中间边,dis[u]+w(u,v)/2+diss[v]的最小值为解。
- #include<stdio.h>
- #include<string.h>
- #include<vector>
- #include<deque>
- #include<map>
- #include<string>
- #define MAX 100005
- #define MAXX 500005
- #define INF 10000000000000000
- using namespace std;
- struct Node{
- int v,w;
- }node;
- vector <Node> edge[MAX],redge[MAX];
- map<string,int> mp;
- long long dis[MAX],diss[MAX],b[MAX],u[MAXX],v[MAXX],w[MAXX];
- char s1[MAX],s2[MAX];
- int n;
- void spfa1(int k)
- {
- int i;
- deque<int> q;
- for(i=;i<=n;i++){
- dis[i]=INF;
- }
- memset(b,,sizeof(b));
- b[k]=;
- dis[k]=;
- q.push_back(k);
- while(q.size()){
- int u=q.front();
- for(i=;i<edge[u].size();i++){
- int v=edge[u][i].v;
- long long w=edge[u][i].w;
- if(dis[v]>dis[u]+w){
- dis[v]=dis[u]+w;
- if(b[v]==){
- b[v]=;
- if(dis[v]>dis[u]) q.push_back(v);
- else q.push_front(v);
- }
- }
- }
- b[u]=;
- q.pop_front();
- }
- }
- void spfa2(int k)
- {
- int i;
- deque<int> q;
- for(i=;i<=n;i++){
- diss[i]=INF;
- }
- memset(b,,sizeof(b));
- b[k]=;
- diss[k]=;
- q.push_back(k);
- while(q.size()){
- int u=q.front();
- for(i=;i<redge[u].size();i++){
- int v=redge[u][i].v;
- long long w=redge[u][i].w;
- if(diss[v]>diss[u]+w){
- diss[v]=diss[u]+w;
- if(b[v]==){
- b[v]=;
- if(diss[v]>diss[u]) q.push_back(v);
- else q.push_front(v);
- }
- }
- }
- b[u]=;
- q.pop_front();
- }
- }
- int main()
- {
- int m,bg,ed,t,i,j;
- while(~scanf("%d%d",&n,&m)){
- t=;
- for(i=;i<=n;i++){
- edge[i].clear();
- redge[i].clear();
- mp.clear();
- }
- memset(u,,sizeof(u));
- memset(v,,sizeof(v));
- memset(w,,sizeof(w));
- for(i=;i<=m;i++){
- scanf(" %s%s%lld",s1,s2,&w[i]);
- if(!mp[s1]) mp[s1]=++t;
- if(!mp[s2]) mp[s2]=++t;
- u[i]=mp[s1];
- v[i]=mp[s2];
- node.v=mp[s2];
- node.w=w[i];
- edge[mp[s1]].push_back(node);
- node.v=mp[s1];
- redge[mp[s2]].push_back(node);
- }
- scanf(" %s%s",s1,s2);
- if(!mp[s1]) mp[s1]=++t;
- if(!mp[s2]) mp[s2]=++t;
- bg=mp[s1],ed=mp[s2];
- spfa1(bg);
- spfa2(ed);
- long long min=INF;
- for(i=;i<=m;i++){
- if(dis[u[i]]==INF||dis[v[i]]==INF) continue;
- if(dis[u[i]]+diss[v[i]]+w[i]/<min) min=dis[u[i]]+diss[v[i]]+w[i]/;
- }
- if(min==INF) printf("-1\n");
- else printf("%lld\n",min);
- }
- return ;
- }
HDU - 3499 Flight 双向SPFA+枚举中间边的更多相关文章
- HDU 3499 Flight spfa+dp
Flight Time Limit : 20000/10000ms (Java/Other) Memory Limit : 65535/65535K (Java/Other) Total Subm ...
- hdu 3499 Flight (最短路径)
Flight Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Su ...
- hdu 3499 flight 【分层图】+【Dijkstra】
<题目链接> 题目大意: 现在给你一些点,这些点之间存在一些有向边,每条边都有对应的边权,有一次机会能够使某条边的边权变为原来的1/2,求从起点到终点的最短距离. 解题分析: 分层图最短路 ...
- BZOJ 1726 [Usaco2006 Nov]Roadblocks第二短路:双向spfa【次短路】
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1726 题意: 给你一个无向图,求次短路. 题解: 两种方法. 方法一: 一遍spfa,在s ...
- 【NOIP2009 T3】 最佳贸易 (双向SPFA)
C 国有 n 个大城市和 m 条道路,每条道路连接这 n 个城市中的某两个城市.任意两个城市之间最多只有一条道路直接相连.这 m 条道路中有一部分为单向通行的道路,一部分为双向通行的道路,双向通行的道 ...
- HDU 1401 Solitaire 双向DFS
HDU 1401 Solitaire 双向DFS 题意 给定一个\(8*8\)的棋盘,棋盘上有4个棋子.每一步操作可以把任意一个棋子移动到它周围四个方向上的空格子上,或者可以跳过它四个方向上的棋子(就 ...
- find the longest of the shortest (hdu 1595 SPFA+枚举)
find the longest of the shortest Time Limit: 1000/5000 MS (Java/Others) Memory Limit: 32768/32768 ...
- HDU - 3499 -(Dijkstra变形+枚举边)
Recently, Shua Shua had a big quarrel with his GF. He is so upset that he decides to take a trip to ...
- Flight HDU - 3499 (分层最短路)
Recently, Shua Shua had a big quarrel with his GF. He is so upset that he decides to take a trip to ...
随机推荐
- Tomcat学习笔记【1】--- WEB服务器、JavaEE、Tomcat背景、Tomcat版本
本文主要讲学习Tomcat需要知道的基础知识. 一 Web服务器 1.1 简介 Web服务器可以解析HTTP协议.当Web服务器接收到一个HTTP请求,会返回一个HTTP响应,例如送回一个HTML页面 ...
- 认识影片版本(CAM、TS、TC、DVD、HD、BD、TVRIP等)
许多朋友在下载电影的时候, 往往会被各种各样的版本标识弄糊涂,今天把各种版本的缩写收集在一起,希望对大家有所帮助 . 引用: 1.CAM(枪版) CAM 通常是用数码摄像机从电影院盗录.有时会使 ...
- Java互斥语义的实现
锁 对象头(Object Header) HotSpot 虚拟机的对象头包括两部分信息:Mark Word(标记字段)和 Klass Pointer(类型指针) Mark Word 用于存储对象自 ...
- Vue-router进阶、单页面应用(SPA)带来的问题
一 . vue-router 进阶 回顾学过的vue-router,并参考官方文档学习嵌套路由等路由相关知识. 二 . 单页面应用(SPA)带来的问题 1 . 虽然单页面应用有优点 , 但是,如果后端 ...
- URAL - 1297 Palindrome —— 后缀数组 最长回文子串
题目链接:https://vjudge.net/problem/URAL-1297 1297. Palindrome Time limit: 1.0 secondMemory limit: 64 MB ...
- Linux bash shell环境变量以及语法规范
摘自: http://blog.csdn.net/abc_ii/article/details/8762739
- 利用ThinkPHP做项目步骤
ThinkPHP使用规则:约定大于配置 创建入口文件: 1.在ThinkPHP目录下创建一个入口文件index.php 2.访问入口文件的同时系统会自动把对应的应用目录文件Test创建出来 3.打开H ...
- UVALive - 7831 :ACM Tax (主席树求树路径上中位数:LCA+主席树)
题意:给定一棵带权树,Q次询问,每次询问路径上的中位数. 思路:中位数分边数奇偶考虑,当当边数为num=奇时,结果就算路径第num/2+1大,用主席树做即可... (做了几道比较难的主席树,都wa了. ...
- tcp攻击
- zero to one (3)
工具使用 AWVS Acunetix Web Vulnerability Scanner(简称AWVS)是一款知名的Web网络漏洞扫描工具,它通过网络爬虫测试你的网站安全,检测流行安全漏洞. 功能及特 ...