题目描述

小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. 【洛谷 P3804】 【模板】后缀自动机

    题目链接 #include <cstdio> #include <cstring> #include <algorithm> using namespace std ...

  2. 2019年北航OO第4单元(UML)总结

    1 架构设计 经过了接近一学期的程序设计训练,在这一单元的第一次作业中我就非常注重架构的设计,竭力避免像之前一样陷入"第一次作业凑合,第二次作业重构"的不健康的迭代模式.整体上来说 ...

  3. 用JS实现输出两个数范围内的随机数

    const rs = require("readline-sync"); function roundNum(min = 0, max = 0) { if (!isNaN(min) ...

  4. mysql 插入一个字段 id自增并设置为主键

    案例 ALTER TABLE customer MODIFY COLUMN custID int NOT NULL AUTO_INCREMENT 开启: ALTER TABLE 表名 MODIFY C ...

  5. ..\USER\stm32f10x.h(428): error: #67: expected a "}" ADC1_2_IRQn = 18, /*!

    MDK软件编译,出现如下错误: ..\USER\stm32f10x.h(428): error: #67: expected a "}" ADC1_2_IRQn = 18, /*! ...

  6. Android笔记(五十五) Android四大组件之一——ContentProvider,使用系统提供的ContentProvider

    因为在Android中,存储系统联系人姓名和电话是存在与不同的ContentProvider中的,具体如何查找,可以从Android的源代码中查看,在android.providers包中列出了所有系 ...

  7. 批处理引擎MapReduce内部原理

    批处理引擎MapReduce内部原理 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.MapReduce作业生命周期 MapReduce作业作为一种分布式应用程序,可直接运行在H ...

  8. MySQL数据库机房裁撤问题总结

    背景:公司某一机房需要裁撤,涉及到大量DB服务器,需要在裁撤截止日期以前完成业务的平滑迁移和设备退还工作. 历时2个多月,经历了设备梳理.裁撤资源评估.裁撤资源申请.裁撤DB部署.裁撤DB业务关系梳理 ...

  9. 第五次个人作业——Alpha测试

    这个作业属于哪个课程 课程链接 这个作业要求在哪里 作业要求 团队名称 巧克力王子与六个小矮人 一.测试项目博客地址 项目名 团队名 博客地址 项目发布地址 西柚排课王 西柚排课王 https://w ...

  10. android在主线程下载文件

    android在主线程下载文件 加入以下代码即可if (android.os.Build.VERSION.SDK_INT > 9) { StrictMode.ThreadPolicy polic ...