化简:够简的了.....但是!翻译绝对有锅。

这个最短路是从n到每个点的单源最短路,也就是最短路径树。

那么,思路就很明确了。建两个图,然后跑两边SPFA,记录下最短路径。

然后,对于两点之间的边,如果最短路不经过它,那么最终图边权+1;

然后在最终图上(边权为0,1,2)跑一遍SPFA即可。

一开始我想复杂了,在想怎么记录路径,怎么重构图.balabala。

然后发现,怎么才能让两点不在最短路径上呢?

SPFA的松弛操作,依据是三角不等式。于是,如果两点之间的最短路的距离如果不等于边权(也就是最短路径不过它俩之间的边)那么就它就是一条会报警的边。

$$看来SPFA最重要的是三角不等式$$

于是,只要暴力跑三遍SPFA即可。

用尽浑身解数,信仰SPFA,各种常数优化,读入挂,我还是没能跑到最优解的第一面....在第二页前几个徘徊.....

代码:(这么长的图论题也不多见,但是好像逛公园就那么长)

#include<iostream>
#include<cstdio>
#include<queue>
#define rg register
using namespace std;
const int maxn=1e6+;
int n,m;
inline int read()
{
int x=,f=;char s=getchar();
while(s>''||s<''){if(s=='-')f=-;s=getchar();}
while(s<=''&&s>=''){x=x*+s-'';s=getchar();}
return x*f;
}
struct edge
{
int to,next,dis;
}e1[maxn],e2[maxn],e[maxn];
int head[maxn],head1[maxn],head2[maxn];
int cnt1,cnt2,cnt;
inline void addedge1(int from,int to,int dis)
{
e1[++cnt1].next=head1[from];
e1[cnt1].to=to;
e1[cnt1].dis=dis;
head1[from]=cnt1;
}
inline void addedge2(int from,int to,int dis)
{
e2[++cnt2].next=head2[from];
e2[cnt2].to=to;
e2[cnt2].dis=dis;
head2[from]=cnt2;
}
inline void addedge(int from,int to,int dis)
{
e[++cnt].next=head[from];
e[cnt].to=to;
e[cnt].dis=dis;
head[from]=cnt;
} int dis1[maxn],vis1[maxn],pre1[maxn];
struct cmp1
{
bool operator () (int a,int b)
{
return dis1[a]>dis1[b];
}
};
inline void spfa1()
{
priority_queue < int , vector < int > , cmp1 > q;
for(rg int i=;i<=n;i++)
{
dis1[i]=;
vis1[i]=;
}
q.push(n);
dis1[n]=;
vis1[n]=;
while(!q.empty())
{
int u=q.top();
q.pop();
vis1[u]=;
for(rg int i=head1[u];i;i=e1[i].next)
{
int v=e1[i].to;
if(dis1[v]>dis1[u]+e1[i].dis)
{
dis1[v]=dis1[u]+e1[i].dis;
if(vis1[v]==)
{
vis1[v]=;
q.push(v);
}
}
}
}
}
int dis2[maxn],vis2[maxn],pre2[maxn];
struct cmp2
{
bool operator () (int a,int b)
{
return dis2[a]>dis2[b];
}
};
inline void spfa2()
{
priority_queue < int , vector < int > , cmp2 > q;
for(rg int i=;i<=n;i++)
{
dis2[i]=;
vis2[i]=;
}
q.push(n);
dis2[n]=;
vis2[n]=;
while(!q.empty())
{
int u=q.top();
q.pop();
vis2[u]=;
for(rg int i=head2[u];i;i=e2[i].next)
{
int v=e2[i].to;
if(dis2[v]>dis2[u]+e2[i].dis)
{
dis2[v]=dis2[u]+e2[i].dis;
if(vis2[v]==)
{
vis2[v]=;
q.push(v);
}
}
}
}
}
int dis[maxn],vis[maxn];
struct cmp
{
bool operator () (int a,int b)
{
return dis[a]>dis[b];
}
};
inline void rebuild()
{
for(rg int i=;i<=n;i++)
{
for(int j=head1[i];j;j=e1[j].next)
{
int v1=e1[j].to;
int v2=e2[j].to;
//cout<<i<<' '<<v1<<endl;
int d=;
if(dis1[v1]-dis1[i]!=e1[j].dis)
d++;
if(dis2[v2]-dis2[i]!=e2[j].dis)
d++;
addedge(v1,i,d);
}
}
}
inline void spfa()
{
priority_queue < int , vector < int > , cmp > q;
for(rg int i=;i<=n;i++)
{
dis[i]=;
vis[i]=;
}
q.push();
dis[]=;
vis[]=;
while(!q.empty())
{
int u=q.top();
q.pop();
vis[u]=;
for(rg int i=head[u];i;i=e[i].next)
{
int v=e[i].to;
if(dis[v]>dis[u]+e[i].dis)
{
dis[v]=dis[u]+e[i].dis;
if(vis[v]==)
{
vis[v]=;
q.push(v);
}
}
}
}
}
int main()
{
n=read();
m=read();
for(rg int i=;i<=m;i++)
{
int a=read(),b=read(),c=read(),d=read();
//scanf("%d%d%d%d",&a,&b,&c,&d);
addedge1(b,a,c);
addedge2(b,a,d);
//addedge(a,b,0);
}
spfa1();
spfa2();
rebuild();
spfa();
printf("%d",dis[n]);
return ;
}

(完)

P3106 [USACO14OPEN]GPS的决斗(最短路)的更多相关文章

  1. Luogu P3106 [USACO14OPEN]GPS的决斗Dueling GPS's(最短路)

    P3106 [USACO14OPEN]GPS的决斗Dueling GPS's 题意 题目描述 Farmer John has recently purchased a new car online, ...

  2. BZOJ 3538 == 洛谷 P3106 [USACO14OPEN]GPS的决斗Dueling GPS's

    P3106 [USACO14OPEN]GPS的决斗Dueling GPS's 题目描述 Farmer John has recently purchased a new car online, but ...

  3. [USACO14OPEN]GPS的决斗Dueling GPS's

    题目概况 题目描述 给你一个\(N\)个点的有向图,可能有重边. 有两个\(GPS\)定位系统,分别认为经过边\(i\)的时间为\(P_i\),和\(Q_i\). 每走一条边的时候,如果一个系统认为走 ...

  4. 洛谷 3106 [USACO14OPEN]GPS的决斗Dueling GPS's 3720 [AHOI2017初中组]guide

    [题解] 这两道题是完全一样的. 思路其实很简单,对于两种边权分别建反向图跑dijkstra. 如果某条边在某一种边权的图中不是最短路上的边,就把它的cnt加上1.(这样每条边的cnt是0或1或2,代 ...

  5. 2018.07.22 洛谷P3106 GPS的决斗Dueling GPS's(最短路)

    传送门 图论模拟题. 这题直接写3个(可以压成一个)spfa" role="presentation" style="position: relative;&q ...

  6. USACO Dueling GPS's

    洛谷 P3106 [USACO14OPEN]GPS的决斗Dueling GPS's 洛谷传送门 JDOJ 2424: USACO 2014 Open Silver 2.Dueling GPSs JDO ...

  7. CSP-S 2019图论总结

    CSP-S 2019图论总结 一.最短路问题 模板 Floyd算法 void floyd() { memset(map,0x3f,sizeof(map)); for(int i=1;i<=n;i ...

  8. [USACO14OPEN] Dueling GPS's[最短路建模]

    题目描述 Farmer John has recently purchased a new car online, but in his haste he accidentally clicked t ...

  9. [BZOJ3538]坑爹的GPS

    题目描述 Description 有一天,\(FJ\) 买了一辆车,但是,他一手下载了两个\(GPS\) 系统.好了现在麻烦的事情来了,\(GPS\) 有一个功能大概大家也知道,如果\(FJ\) 没有 ...

随机推荐

  1. css父元素透明度(opacity)对子元素的影响

    首先子元素会继承父元素的透明度: 设置父元素opacity:0.5,子元素不设置opacity,子元素会受到父元素opacity的影响,也会有0.5的透明度. 其次子元素的透明度是基于父元素的透明度计 ...

  2. SpringCloud系列-整合Hystrix的两种方式

    Hystrix [hɪst'rɪks],中文含义是豪猪,因其背上长满棘刺,从而拥有了自我保护的能力.本文所说的Hystrix是Netflix开源的一款容错框架,同样具有自我保护能力. 本文目录 一.H ...

  3. Vue中的循环以及修改差值表达式

    0828自我总结 一.Vue中的循环 v-for 常见的4总情况 #第一种 <div v-for="item in items"></div> #第二种 & ...

  4. VirtualBox for Mac 6.0.14 开源免费虚拟机方案

    VirtualBox for mac是一款开源虚拟机软件,你可以利用该软件在Mac OS平台上运行Windows软件,即可以在一定程度上弥补Mac OS平台软件不足的劣势,玩家也可以获得Windows ...

  5. Redis Getshell总结

    Redis默认端口是6379,可以通过zoomeye或者shodan来查找开放6379的端口,能收到很多对外开放6379端口的ip,而且绝大部门是未授权访问的,因为默认redis是未授权访问.但是还有 ...

  6. powershell 基础

    目录 本教程概述 用到的工具 标签 简介 0x01使用简介 0x02脚本编写 0x03实例讲解 本教程概述 本课我们学习powershell使用. 用到的工具 cmd.exe   powershell ...

  7. USACO环绕岛屿Surround the Islands 并查集 枚举暴力

    题目描述 Farmer John has bought property in the Caribbean and is going to try to raise dairy cows on a b ...

  8. Cocos2d-x 学习笔记(9) Action 运行原理

    1. 从一个Action开始 1.1 创建 在Scene里写一个Sprite,并添加Action: Sprite *sp = Sprite::create("m1.png"); M ...

  9. pycharm在进行debug时不小心把console关闭了,恢复console的办法

    点击下图中右边的箭头就恢复了 此时可看到console已恢复

  10. C++bosst遍历文件目录,根据文件名返回文件路径。

    VS2071安装Boost库 安装boost库 接着安装boost_system-vc140(可根据开发需求,更改版本) 废话不多说,上代码 // 测试程序.cpp : 此文件包含 "mai ...