【NOIP2017】逛公园 最短路+DP
诶,去年场上不会处理$0$的环,只拿了$60$有点可惜。
我们先不管边边权为$0$的边。
我们先跑一次最短路,令$dis[u]$表示从$1$至$u$的最短路的长度。
那么根据题目的要求,从起点走到$u$号点的路径长度只可能在区间$[dis[u],dis[u]+k]$中。
令$f[i][j]$表示当前从起点走到$i$,行走的路程为$dis[i]+j$的方案数。
不妨发现这个东西可以通过类似分层图最短路的方式进行更新,然后就直接更新就行了。
然而这一题中有部分点存在边权为$0$的边,一旦走入一个$0$环的话采用上述的方法会$TLE$。
于是我们把上面的通过分层图最短路的更新方式,更换为记忆化搜索,我们在当前搜索的路径上打上标记,然后若走到之前标记过的点,那么直接输出$-1$退出即可。这种方式可以有效避免将无需经过的零环纳入考虑范围内。
(可能是个人写法的原因),经过$1$的零环需要特判,直接在最短路里判掉就好了。
然后就没了。时间复杂度:$O(mk+m log n)$。
#include<bits/stdc++.h>
#define M 100005
using namespace std; struct edge{int u,v,next;}e[M*]={}; int head[M]={},head1[M]={},use=;
void add(int x,int y,int z){use++;e[use].u=y;e[use].v=z;e[use].next=head[x];head[x]=use;}
void add1(int x,int y,int z){use++;e[use].u=y;e[use].v=z;e[use].next=head1[x];head1[x]=use;}
int n,m,k,MOD; struct node{
int x,dis;
node(){x=dis=;}
node(int xx,int ddis){x=xx; dis=ddis;}
friend bool operator <(node a,node b){return a.dis>b.dis;}
}a[M];
priority_queue<node> q;
int vis[M]={},dis[M]={},v[M][]={},f[M][]={},in[M][]={};
int dij(){
q.push(node(,));
while(!q.empty()){
node U=q.top(); q.pop();
int u=U.x;
if(vis[u]) continue;
vis[u]=; dis[u]=U.dis;
for(int i=head[u];i;i=e[i].next){
if(dis[u]+e[i].v==&&e[i].u==) return ;
if(!vis[e[i].u]) q.push(node(e[i].u,dis[u]+e[i].v));
}
}
return ;
} int dfs(int x,int p){
if(f[x][p]!=-) return f[x][p];
if(in[x][p]) return -;
int res=; in[x][p]=;
for(int i=head1[x];i;i=e[i].next){
int v=dis[x]-dis[e[i].u]+p-e[i].v;
if(<=v&&v<=k){
int now=dfs(e[i].u,v);
if(now==-) return -;
res=(res+now)%MOD;
}
}
f[x][p]=res; in[x][p]=;
return res;
} int Main(){
memset(vis,,sizeof(vis)); memset(dis,,sizeof(dis));
memset(f,-,sizeof(f)); memset(e,,sizeof(e));
memset(head,,sizeof(head)); memset(head1,,sizeof(head1)); use=;
memset(in,,sizeof(in));
memset(v,,sizeof(v));
scanf("%d%d%d%d",&n,&m,&k,&MOD);
for(int i=;i<=m;i++){
int x,y,z; scanf("%d%d%d",&x,&y,&z);
add(x,y,z);
add1(y,x,z);
}
if(dij()==){printf("-1\n"); return ;}
int ans=; f[][]=;
for(int i=;i<=k;i++){
int now=dfs(n,i);
if(now==-){
printf("-1\n");
return ;
}
ans=(ans+now)%MOD;
}
printf("%d\n",ans);
} int main(){
int cas; cin>>cas;
while(cas--) Main();
}
【NOIP2017】逛公园 最短路+DP的更多相关文章
- [NOIP2017]逛公园 最短路+拓扑排序+dp
题目描述 给出一张 $n$ 个点 $m$ 条边的有向图,边权为非负整数.求满足路径长度小于等于 $1$ 到 $n$ 最短路 $+k$ 的 $1$ 到 $n$ 的路径条数模 $p$ ,如果有无数条则输出 ...
- [NOIP2017]逛公园 最短路图 拓扑序DP
---题面--- 题解: 挺好的一道题. 首先我们将所有边反向,跑出n到每个点的最短路,然后f[i][j]表示从i号节点出发,路径长比最短路大j的方案数. 观察到,如果图中出现了0环,那么我们可以通过 ...
- [NOIP2017] 逛公园 解题报告(DP)
我很不想说 在我的AC代码上我打了表,但实在没有办法了.莫名的8,9个点RE.然而即便是打表...也花了我很久. 这大概是NOIP2017最难的题了,为了让不懂的人更容易理解,这篇题解会比较详细 我的 ...
- [NOIP2017] 逛公园 (最短路,动态规划&记忆化搜索)
题目链接 Solution 我只会60分暴力... 正解是 DP. 状态定义: \(f[i][j]\) 代表 \(1\) 到 \(i\) 比最短路长 \(j\) 的方案数. 那么很显然最后答案也就是 ...
- 洛谷 P3953 [ NOIP 2017 ] 逛公园 —— 最短路DP
题目:https://www.luogu.org/problemnew/show/P3953 主要是看题解...还是觉得好难想啊... dfs DP,剩余容量的损耗是边权减去两点最短路差值...表示对 ...
- 【题解】NOIP2017逛公园(DP)
[题解]NOIP2017逛公园(DP) 第一次交挂了27分...我是不是必将惨败了... 考虑这样一种做法,设\(d_i\)表示从该节点到n节点的最短路径,\(dp(i,k)\)表示从\(i\)节点 ...
- [NOIP2017] 逛公园
[NOIP2017] 逛公园 题目大意: 给定一张图,询问长度 不超过1到n的最短路长度加k 的1到n的路径 有多少条. 数据范围: 点数\(n \le 10^5\) ,边数\(m \le 2*10^ ...
- 【比赛】NOIP2017 逛公园
考试的时候灵光一闪,瞬间推出DP方程,但是不知道怎么判-1,然后?然后就炸了. 后来发现,我只要把拓扑和DP分开,中间加一个判断,就AC了,可惜. 看这道题,我们首先来想有哪些情况是-1:只要有零环在 ...
- NOIP2017逛公园(dp+最短路)
策策同学特别喜欢逛公园.公园可以看成一张N个点M条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口,N号点是公园的出口,每条边有一个非负权值, 代表策策经过这条边所要花的时间. 策策每天都会 ...
随机推荐
- SQLServerException:将截断字符串或二进制数据的解决方法
SQLServerException:将截断字符串或二进制数据的解决方法: 最近使用JPA进行保存对象到数据库中怎么也添加不进去,始终报错 主要原因就是你增加的数据字段长度超过数据库中字段所定义长度, ...
- 2018.09.26 bzoj5221: [Lydsy2017省队十连测]偏题(数学推导+矩阵快速幂)
传送门 由于没有考虑n<=1的情况T了很久啊. 这题很有意思啊. 考试的时候根本不会,骗了30分走人. 实际上变一个形就可以了. 推导过程有点繁杂. 直接粘题解上的请谅解. 不得不说这个推导很妙 ...
- 安装memcache服务
d:\tools\memcache\setup\memcached -d install
- Map的常用操作
public static void main(String[] args) { Map<String, String> map = new HashMap<>(); map. ...
- 优秀前端工程师必备: 非常常用的checkbox的骚操作---全选和单选demo
提要: 前端开发的时候, 经常会遇到表格勾选, 单个勾选判断是否全选的事情.趁着有时间, 总结一下以备不时之需! 就像下面这个栗子: 1 源代码: h5 // 全选框 <input type=& ...
- HDU 1061 Rightmost Digit (快速幂取模)
题意:给定一个数,求n^n的个位数. 析:很简单么,不就是快速幂么,取余10,所以不用说了,如果不会快速幂,这个题肯定是周期的, 找一下就OK了. 代码如下: #include <iostrea ...
- CentOS7中实用的命令总结
一:软件安装配置方面 这里总结一下对个人很实用的功能 1.查询rpm包的依赖哪些文件:rpm -qpR xx.rpm,然后rpm会分析此包的文件依赖性 2.查询系统中安装了哪些rpm包:rpm -qa ...
- C/C++的Name Mangling
C语言 函数 1.void __CALLTYPE f();2.int __CALLTYPE f();3.int __CALLTYPE f(int);4.double __CALLTYPE f(int, ...
- Java火焰图在Netflix的实践
转自 http://www.infoq.com/cn/news/2015/08/java-flamegraph 亲爱的读者:我们最近添加了一些个人消息定制功能,您只需选择感兴趣的技术主题,即可获取重要 ...
- Android操作HTTP实现与服务器通信
(转自http://www.cnblogs.com/hanyonglu/archive/2012/02/19/2357842.html) 本示例以Servlet为例,演示Android与Servlet ...