题面

这道题是一道不错的计数类DP;

首先我们一定要跑一遍dijkstra来求得每个点到1号点的最短路;

注意题干,题中并没有说所有点都可以到达n好点,只说了存在一条1号点到n号点的路径;所以我们在反向图上BFS求出那些点可以到达n号点;

然后就是dp。我们设计状态:f[u][rest]表示到达u号点时,还可以走额外rest这么长的路;很显然得:f[u][rest]=sigma(f[v][rest-w+dis[v]-dis[u]])

其中dis[u]表示从1到u得最短路;w表示从u到v的边权;

为了在图中DP方便,可以采用记忆化搜索来进行记数;

判断是否输出-1的方法:如果这个状态在栈里,那么输出-1;表示存在0环;

特别注意:当判断-1时,你可能返回-INT_MAX来表示结束程序,但这道题的模数比较小,所以一定不能在返回-INT_MAX时把这个数取模,否则会输出一个负数(不是-INT_MAX),让程序继续进行,并不能输出-1;

#include <bits/stdc++.h>
const int INF=0x7FFFFFFF;
using namespace std;
struct littlestar{
int to;
int nxt;
int w;
}star[400010],star2[400010];
int head[400010],cnt,head2[400010],cnt2;
void add(int u,int v,int w)
{
star[++cnt].to=v;
star[cnt].nxt=head[u];
head[u]=cnt;
star[cnt].w=w;
}
void add2(int u,int v,int w)
{
star2[++cnt2].to=v;
star2[cnt2].nxt=head2[u];
star2[cnt2].w=w;
head2[u]=cnt2;
}
int n,m,k,p;
int dis[100010],vis[100010],bo[100010];
int ans[100010][51];
int st[100010][51];
inline void pre()
{
memset(bo,0,sizeof(bo));
memset(ans,-1,sizeof(ans));
memset(st,0,sizeof(st));
memset(vis,0,sizeof(vis));
memset(head,0,sizeof(head));
memset(head2,0,sizeof(head2));
cnt=0;
cnt2=0;
}
priority_queue<pair<int,int> > q;
void dijkstra()
{
q.push(make_pair(0,1));
dis[1]=0;
while(q.size()){
int u=q.top().second;
q.pop();
if(vis[u]) continue;
vis[u]=1;
for(register int i=head[u];i;i=star[i].nxt){
int v=star[i].to;
if(dis[v]>dis[u]+star[i].w){
dis[v]=dis[u]+star[i].w;
q.push(make_pair(-dis[v],v));
}
}
}
}
queue<int> q2;
void panduan()
{
q2.push(n);
bo[n]=1;
while(q2.size()){
int u=q2.front();
q2.pop();
for(register int i=head2[u];i;i=star2[i].nxt){
int v=star2[i].to;
if(bo[v]==0){
q2.push(v);
bo[v]=1;
}
}
}
}
int dp(int u,int rest)
{
if(rest<0) return 0;
if(st[u][rest]==1) return -INF;
if(ans[u][rest]!=-1) return ans[u][rest];
long long tmpans=0;
st[u][rest]=1;
if(u==n) ++tmpans;
for(register int i=head[u];i;i=star[i].nxt){
int v=star[i].to;
if(bo[v]==0) continue;
int ha=dis[v]-dis[u];
int tmp=dp(v,rest-(star[i].w-ha));
if(tmp==-INF){
return -INF;
}
else{
tmpans=(tmpans+tmp)%p;
}
}
ans[u][rest]=tmpans%p;
st[u][rest]=0;
return tmpans;
}
int main()
{
int t;
cin>>t;
while(t--){
pre();
cin>>n>>m>>k>>p;
for(register int i=1;i<=m;i++){
int u,v,w;
scanf("%d%d%d",&u,&v,&w);
add(u,v,w);
add2(v,u,w);
}
for(int i=1;i<=n;i++) dis[i]=INF;
dijkstra();
panduan();
long long w=dp(1,k);
if(w==-INF){
cout<<"-1"<<endl;
}
else{
cout<<w<<endl;
}
}
}

NOIP 2017 逛公园 题解的更多相关文章

  1. [NOIp 2017]逛公园

    Description 策策同学特别喜欢逛公园.公园可以看成一张$N$个点$M$条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口,$N$号点是公园的出口,每条边有一个非负权值, 代表策策经 ...

  2. NOIP 2017 逛公园 记忆化搜索 最短路 好题

    题目描述: 策策同学特别喜欢逛公园.公园可以看成一张N个点MM条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口,N号点是公园的出口,每条边有一个非负权值, 代表策策经过这条边所要花的时间. ...

  3. 洛谷 P3953 [ NOIP 2017 ] 逛公园 —— 最短路DP

    题目:https://www.luogu.org/problemnew/show/P3953 主要是看题解...还是觉得好难想啊... dfs DP,剩余容量的损耗是边权减去两点最短路差值...表示对 ...

  4. NOIP 2017 逛公园 - 动态规划 - 最短路

    题目传送门 传送门 题目大意 给定一个$n$个点$m$条边的带权有向图,问从$1$到$n$的距离不超过最短路长度$K$的路径数. 跑一遍最短路. 一个点拆$K + 1$个点,变成一个DAG上路径计数问 ...

  5. NOIP2017 逛公园 题解报告 【最短路 + 拓扑序 + dp】

    题目描述 策策同学特别喜欢逛公园.公园可以看成一张NNN个点MMM条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口,NNN号点是公园的出口,每条边有一个非负权值, 代表策策经过这条边所要花 ...

  6. [NOIP2017]逛公园 题解

    我连D1T3都不会我联赛完蛋了 题目描述 策策同学特别喜欢逛公园.公园可以看成一张 N 个点 M 条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口, N 号点是公园的出口,每条边有一个非负 ...

  7. 线段树 || BZOJ1756: Vijos1083 小白逛公园 || P4513 小白逛公园

    题面:小白逛公园 题解: 对于线段树的每个节点除了普通线段树该维护的东西以外,额外维护lsum(与左端点相连的最大连续区间和).rsum(同理)和sum……就行了 代码: #include<cs ...

  8. 【NOIP 2017】逛公园

    Description 策策同学特别喜欢逛公园.公园可以看成一张N个点M条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口,N号点是公园的出口,每条边有一个非负权值, 代表策策经过这条边所要 ...

  9. [NOIP 2017 day1]逛公园

    题目描述 策策同学特别喜欢逛公园. 公园可以看成一张 N 个点 M 条边构成的有向图,且没有自环和重边.其中 1 号点是公园的入口, N 号点是公园的出口,每条边有一个非负权值,代表策策经过这条边所要 ...

随机推荐

  1. 【CUDA 基础】3.1 CUDA执行模型概述

    title: [CUDA 基础]3.1 CUDA执行模型概述 categories: CUDA Freshman tags: CUDA SM SIMT SIMD Fermi Kepler toc: t ...

  2. Harmonic Number (LightOJ 1234)(调和级数 或者 区块储存答案)

    题解:隔一段数字存一个答案,在查询时,只要找到距离n最近而且小于n的存答案值,再把剩余的暴力跑一遍就可以. #include <bits/stdc++.h> using namespace ...

  3. codeforces340C

    Tourist Problem CodeForces - 340C Iahub is a big fan of tourists. He wants to become a tourist himse ...

  4. BZOJ 4823 Luogu P3756 [CQOI2017]老C的方块 (网络流、最小割)

    题目链接 (Luogu) https://www.luogu.org/problem/P3756 (BZOJ) http://lydsy.com/JudgeOnline/problem.php?id= ...

  5. Java线程之FutureTask

    简述 FutureTask是Future接口的实现类,并提供了可取消的异步处理的功能,它包含了启动和取消(start and cancel)任务的方法,同时也包含了可以返回FutureTask状态(c ...

  6. selenium.common.exceptions.StaleElementReferenceException: Message: stale element reference: element is not attached to the page document

    抓取网页代码后,由于是在同一个li标签下,所以使用一次性抓取,所有的a标签,然后循环做不同的操作,但是抛出找不到元素异常. def office_page(_chrome: Chrome): sn = ...

  7. python-matplotlib-2

    figure的使用 x = np.linspace(-1, 1, 50) y1 = 2 * x + 1 #figure 1 plt.figure() plt.plot(x, y1) # figure ...

  8. Nginx-rtmp之 ngx_rtmp_send.c 文件分析

    1. 简述 1.1 RTMP 消息类型 /* RTMP message types */ #define NGX_RTMP_MSG_CHUNK_SIZE 1 #define NGX_RTMP_MSG_ ...

  9. C++ string与int的互相转换

    原文地址 C++本身就提供了字符串与整型数之间的互换,那就是利用stringstream.下面是使用方法: 核心: 利用C++中的stringstream流. 由于使用过程比较简单就不再赘述,直接给出 ...

  10. vue 添加一个二级目录

    原先的: http://xxx.xx.xxx.xxx:xxx/#/ 现在的: http://xxx.xx.xxx.xxx:xxx/guangfu/#/