题目描述

小P和小R在玩一款益智游戏。游戏在一个正权有向图上进行。 小P 控制的角色要从A 点走最短路到B 点,小R 控制的角色要从C 点走最短路到D 点。 一个玩家每回合可以有两种选择,移动到一个相邻节点或者休息一回合。 假如在某一时刻,小P 和小R 在相同的节点上,那么可以得到一次特殊奖励,但是在每 个节点上最多只能得到一次。 求最多能获得多少次特殊奖励

输入格式

第一行两个整数n,m 表示有向图的点数和边数。 接下来m 行每行三个整数xi,yi,li,表示从xi 到yi 有一条长度为li 的边。 最后一行四个整数A,B,C,D,描述小P 的起终点,小R 的起终点。

输出格式

输出一个整数表示最多能获得多少次特殊奖励。若小P不能到达B点或者小R不能到达D点则输出-1。

输入样例

5 5
  1 2 1
  2 3 2
  3 4 4
  5 2 3
  5 3 5
  1 3 5 4

输出样例

2
提示
  对于30%的数据,满足n≤50 对于60%的数据,满足n≤1000,m≤5000 对于100%的数据,满足n≤50000,m≤200000,1≤li≤500000000

分析

A到B,C到D的最短路有很多条。

但对于任意一条A到B的最短路和任意一条C到D的最短路如果他们有交点,那交点一定是连续的,否则一定存在交点更多的两条最短路。

如图

如果A到B的路径(红色)与C到D的路径(蓝色)有不连续的交点E与F

那么E通过路径1到达F与E通过路径2到达F的距离应该是一样,否则路径A-B,路径C-D中有一条不是最短路

此时,若A走到达E后走路径2到达F会与路径C-D有更多交点

即走绿色这条路<-也是最短路

所以最优方案中特殊奖励的点一定是连续的。

于是我们可以用同时在A-B最短路上和C-D最短路上的有向边构成一个新图

那么新图一定是一个有向无环图,但不一定联通

最后对新图拓扑排序dp最长链即可

代码(给你们演示一下堆优化的SPFA)

#include<queue>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=;
const int maxm=;
struct node{int id;long long d;};
bool operator <(node a,node b){return a.d>b.d;}
int info[maxn],nx[maxm<<],v[maxm<<],w[maxm<<],re[maxm<<];long long d[][maxn];
int n,m,en,ans,ecnt,p[],dp[maxn],vis[maxn],inp[maxn],top[maxn],lef[maxm<<];priority_queue<node>q;
void add(int u1,int v1,int w1,int r){nx[++ecnt]=info[u1];info[u1]=ecnt;v[ecnt]=v1;w[ecnt]=w1;re[ecnt]=r;}
void SPFA(int k)
{
memset(vis,,sizeof vis);
d[k][p[k]]=;q.push((node){p[k],});
while(!q.empty())
{
node nw=q.top();q.pop();
if(vis[nw.id]==)continue;vis[nw.id]=;d[k][nw.id]=nw.d;
for(int i=info[nw.id];i;i=nx[i])
if(!((k%)^re[i])&&d[k][v[i]]>nw.d+w[i])q.push((node){v[i],nw.d+w[i]});
}
}
int main()
{
scanf("%d%d",&n,&m);memset(d,0x7f,sizeof d);
for(int i=,u1,v1,w1;i<=m;i++)scanf("%d%d%d",&u1,&v1,&w1),add(u1,v1,w1,),add(v1,u1,w1,);
for(int k=;k<=;k++)scanf("%d",&p[k]),SPFA(k);
if(d[][p[]]==d[][]||d[][p[]]==d[][]){puts("-1");return ;}
for(int x=;x<=n;x++)
{
for(int e=info[x];e;e=nx[e])
if(re[e]&&d[][x]+w[e]+d[][v[e]]==d[][p[]]&&d[][x]+w[e]+d[][v[e]]==d[][p[]])
lef[e]=,inp[v[e]]++;
}
for(int x=;x<=n;x++)if(inp[x]==)top[++en]=x,dp[x]=;
for(int i=;i<=en;i++)
{
int nw=top[i];
for(int e=info[nw];e;e=nx[e])
if(lef[e])
{
dp[v[e]]=max(dp[v[e]],dp[nw]+);
ans=max(ans,dp[v[e]]);inp[v[e]]--;
if(!inp[v[e]])top[++en]=v[e];
}
}
printf("%d\n",ans);
}

【CSP模拟赛】益智游戏(最短路(DJSPFA)&拓扑排序)的更多相关文章

  1. Day3 最短路 最小生成树 拓扑排序

    Day3 最短路 最小生成树 拓扑排序 (一)最短路 一.多源最短路 从任意点出发到任意点的最短路 1. Floyd \(O(n^3)\) for(int k=1;k<=n;k++) for(i ...

  2. CSP模拟赛游记

    时间:2019.10.5 考试时间:100分钟(连正式考试时间的一半还没有到)题目:由于某些原因不能公开. 由于第一次接触NOIinux系统所以连怎么建文件夹,调字体,如何编译都不知道,考试的前半小时 ...

  3. [POI2014]RAJ(最短路,拓扑排序)

    对于一个点 \(x\) 如何求答案? 由于这个图是个有向无环图,可以先拓扑排序一遍,求出每个点的拓扑序,从起点到它的最长路 \(d2\),从它到终点的最长路 \(d1\).(我写代码是这么写的,注意顺 ...

  4. 2019河北省大学生程序设计竞赛(重现赛)J-舔狗 (拓扑排序)

    题目链接:https://ac.nowcoder.com/acm/contest/903/J 题意:给你 n 个舔狗和他喜欢的人,让你俩俩配对(只能和喜欢它的和它喜欢的),求剩下的单身狗数量. 思路: ...

  5. 【CSP模拟赛】Freda的迷宫(桥)

    题目描述 Freda是一个迷宫爱好者,她利用业余时间建造了许多迷宫.每个迷宫都是由若干房间和走廊构成的,每条走廊都连接着两个不同的房间,两个房间之间最多只有一条走廊直接相连,走廊都是双向通过.  黄昏 ...

  6. 【csp模拟赛九】--dfs

    思路: 这道题可以宽搜,深搜,最短路 代码: #include<cstdio> #include<cstring> #include<iostream> #incl ...

  7. 纪中某日c组模拟赛 2314. 最短路

    2314. 最短路 (File IO): input:dti.in output:dti.out 时间限制: 1000 ms  空间限制: 262144 KB  具体限制   Goto Problem ...

  8. CSP模拟赛2游记

    这次由于有课迟到30min,了所以只考了70min. 调linux配置调了5min,只剩下65min了. T1:有点像标题统计,但要比他坑一点,而且我就被坑了,写了一个for(int i=1;i< ...

  9. 【NOI P模拟赛】最短路(树形DP,树的直径)

    题面 给定一棵 n n n 个结点的无根树,每条边的边权均为 1 1 1 . 树上标记有 m m m 个互不相同的关键点,小 A \tt A A 会在这 m m m 个点中等概率随机地选择 k k k ...

随机推荐

  1. ubuntu安装mysql数据库方法

    ubuntu基于linux的免费开源桌面PC操作系统,十分契合英特尔的超极本定位,支持x86.64位和ppc架构.一个比较流行的Linux操作系统,不仅简单易用,而且和Windows相容性非常好.那么 ...

  2. JavaScript内置一些方法的实现原理--new关键字,call/apply/bind方法--实现

    先学习下new操作符吧 new关键字调用函数的心路历程: 1.创建一个新对象 2.将函数的作用域赋给新对象(this就指向这个对象) 3.执行函数中的代码 4.返回这个对象 根据这个的思路,来实现一个 ...

  3. Swift 4 中的泛型

    作为Swift中最重要的特性之一,泛型使用起来很巧妙.很多人都不太能理解并使用泛型,特别是应用开发者.泛型最适合libraries, frameworks, and SDKs的开发.在这篇文章中,我将 ...

  4. 【hadoop】在eclipse上运行WordCount的操作过程

    序:本以为今天花点时间将WordCount例子完全理解到,但高估自己了,更别说我只是在大学选修一学期的java,之后再也没碰过java语言了 总的来说,从宏观上能理解具体的程序思路,但具体到每个代码有 ...

  5. 臀部——哑铃&杠铃

  6. python接口自动化11-post传data参数案例

    前言: 前面登录博客园的是传json参数,有些登录不是传json的,如jenkins的登录,本篇以jenkins登录为案例,传data参数. 一.登录jenkins抓包 1.登录jenkins,输入账 ...

  7. python 执行系统命令模块比较

    python 执行系统命令模块比较 1.os.system模块 仅仅在子终端运行命令,返回状态码,0为成功,其他为失败,但是不返回执行结果 如果再命令行下执行,结果直接打印出来 >>> ...

  8. InvenSense 美国公司

    InvenSense为智能型运动处理方案的先驱.全球业界的领导厂商,驱动了运动感测人机接口在消费性电子产品上的应用.公司提供的集成电路(IC)整合了运动传感器-陀螺仪以及相对应的软件,有别于其他厂商, ...

  9. MPU-6050

    MPU-6000(6050)为全球首例整合性6轴运动处理组件,相较于多组件方案,免除了组合陀螺仪与加速器时间轴之差的问题,减少了大量的封装空间.当连接到三轴磁强计时,MPU-60X0提供完整的9轴运动 ...

  10. 《少年先疯队》第八次团队作业:Alpha冲刺第四天

    前言    第四天冲刺会议    时间:2019.6.17    地点:宿舍 4.1 今日完成任务情况以及遇到的问题.   4.1.1今日完成任务情况 姚玉婷:管理员功能模块中,收费管理功能的实现. ...