P3106 [USACO14OPEN]GPS的决斗(最短路)

化简:够简的了.....但是!翻译绝对有锅。
这个最短路是从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的决斗(最短路)的更多相关文章
- Luogu P3106 [USACO14OPEN]GPS的决斗Dueling GPS's(最短路)
P3106 [USACO14OPEN]GPS的决斗Dueling GPS's 题意 题目描述 Farmer John has recently purchased a new car online, ...
- BZOJ 3538 == 洛谷 P3106 [USACO14OPEN]GPS的决斗Dueling GPS's
P3106 [USACO14OPEN]GPS的决斗Dueling GPS's 题目描述 Farmer John has recently purchased a new car online, but ...
- [USACO14OPEN]GPS的决斗Dueling GPS's
题目概况 题目描述 给你一个\(N\)个点的有向图,可能有重边. 有两个\(GPS\)定位系统,分别认为经过边\(i\)的时间为\(P_i\),和\(Q_i\). 每走一条边的时候,如果一个系统认为走 ...
- 洛谷 3106 [USACO14OPEN]GPS的决斗Dueling GPS's 3720 [AHOI2017初中组]guide
[题解] 这两道题是完全一样的. 思路其实很简单,对于两种边权分别建反向图跑dijkstra. 如果某条边在某一种边权的图中不是最短路上的边,就把它的cnt加上1.(这样每条边的cnt是0或1或2,代 ...
- 2018.07.22 洛谷P3106 GPS的决斗Dueling GPS's(最短路)
传送门 图论模拟题. 这题直接写3个(可以压成一个)spfa" role="presentation" style="position: relative;&q ...
- USACO Dueling GPS's
洛谷 P3106 [USACO14OPEN]GPS的决斗Dueling GPS's 洛谷传送门 JDOJ 2424: USACO 2014 Open Silver 2.Dueling GPSs JDO ...
- CSP-S 2019图论总结
CSP-S 2019图论总结 一.最短路问题 模板 Floyd算法 void floyd() { memset(map,0x3f,sizeof(map)); for(int i=1;i<=n;i ...
- [USACO14OPEN] Dueling GPS's[最短路建模]
题目描述 Farmer John has recently purchased a new car online, but in his haste he accidentally clicked t ...
- [BZOJ3538]坑爹的GPS
题目描述 Description 有一天,\(FJ\) 买了一辆车,但是,他一手下载了两个\(GPS\) 系统.好了现在麻烦的事情来了,\(GPS\) 有一个功能大概大家也知道,如果\(FJ\) 没有 ...
随机推荐
- TouchListener PK OnTouchEvent + 多点触碰
1.基于监听的TouchListener 代码示例: 实现效果图: 实现代码: main.xml <RelativeLayout xmlns:android="http://schem ...
- ES(Elastic Search)update操作设置无 docment时进行insert
最近使用一套数据加工中间工具,查看es操作中的update操作.其中方法命名为updateOrInsert.但是没发现代码中有ES的insert方法调用.于是仔细分析了代码逻辑. 经过一路追溯,直至E ...
- python-setup模块
本地打包,setup安装 一.distutils 使用:distutils 进行打包,步骤如下,以单一文件为例. 1.创建文件 在同一目录下.写一个foo.py文件: #-*- coding:utf- ...
- nginx的负载均衡实战
前言 nginx是一个高性能的HTTP和反向代理的服务器.它有三个最基本的功能,一是当做web服务器.二是作为反向代理服务器.三是提供负载均衡(在反向代理基础上),由于它占有内存小,并发能力强,所以在 ...
- MYSQL事件隔离级别以及复读,幻读,脏读的理解
一.mysql事件隔离级别 1未提交读(READUNCOMMITTED) 另一个事务修改了数据,但尚未提交,而本事务中的SELECT会读到这些未被提交的数据(脏读)( 隔离级别最低,并发性能高 ) 2 ...
- 微信小程序路由跳转
微信小程序路由跳转 1.wx.switchTab(Object object) 这里的tabBar是底下的导航栏指定的页面, 跳转到 tabBar 页面,并关闭其他所有非 tabBar 页面 参数 O ...
- Dubbo+Zookeeper(一)Zookeeper初识
前面花了一段时间去学习SpringCloud的相关知识,主要是理解微服务的概念并使用SpringCloud的一系列组件实现微服务落地.学习这些组件本身是简单的,跟着操作一遍基本就会了,这也得益于Spr ...
- VSCode 安装 code 命令
VSCode 提供 code 命令直接从命令行中打开文件目录,此时需要先安装 code 命令. 1.首先打开 VSCode 2.使用 command + shift + p (注意window 下使用 ...
- python中list切片详解
语法:[start:stop:step] step代表切片步长:切片区间为[start,stop),包含start但不包含stop 1.step > 0,从左往右切片 2.step <0, ...
- ThreadPoolExecutor使用方法
先看构造方法 ,ThreadPoolExecutor共4个构造方法: 直接看参数最多的7个参数分别代表: public ThreadPoolExecutor(int corePoolSize, int ...