洛谷 P3994 高速公路
https://www.luogu.org/problemnew/show/P3994
设dp[i] 表示第i个城市到根节点的最小花费
dp[i]=min{ (dis[i]-dis[j])*P[i]+Q[i]+dp[j] }
这是O(n^2)的
这个式子可以斜率优化
dp[i]+dis[j]*P[i]=dis[i]*P[i]+Q[i]+dp[j]
就是一条斜率为P[i]的直线,截(dis[j],dp[j])的最小截距
在根往下走的过程中,斜率单调递增
这就体现了 为什么题目中说“i号城市是j号城市的某个祖先,那么一定存在Pi<=Pj”
我们按dfs序dp
现在唯一的问题就是如何得到 一个点到根节点路径上的单调队列
只需要考虑如何去除兄弟节点的子树对单调队列的影响
即在一个节点退出dfs时,将单调队列恢复为这个节点开始dfs的情况
头指针只是不断的+1,没有涉及到单调队列中元素的修改,所以记录下头指针在哪个位置即可
尾指针涉及到元素的替换,但是它只会替换一个元素,所以记录下尾指针的位置,以及被当前点替换的元素是谁
当节点退出dfs时,恢复记录的这三个值即可
这样的话,一个节点多次出队入队,时间复杂度就不是O(n)了
所以二分出队位置,时间复杂度为O(nlogn)
朴素的DP:
#include<cstdio>
#include<iostream>
#include<algorithm> using namespace std; #define N 1000001 typedef long long LL; int P[N],Q[N]; int front[N],to[N<<],nxt[N<<],val[N<<],tot; int fa[N]; LL dis[N]; int t;
LL mi[N]; void read(int &x)
{
x=; char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) { x=x*+c-''; c=getchar(); }
} void add(int u,int v,int w)
{
to[++tot]=v; nxt[tot]=front[u]; front[u]=tot; val[tot]=w;
to[++tot]=u; nxt[tot]=front[v]; front[v]=tot; val[tot]=w;
} void dfs(int x,int f)
{
for(int i=front[x];i;i=nxt[i])
{
if(to[i]==f) continue;
dis[to[i]]=dis[x]+val[i];
mi[to[i]]=dis[to[i]]*P[to[i]]+Q[to[i]];
t=fa[to[i]];
while(t!=)
{
mi[to[i]]=min(mi[to[i]],(dis[to[i]]-dis[t])*P[to[i]]+Q[to[i]]+mi[t]);
t=fa[t];
}
dfs(to[i],x);
}
} int main()
{
int n,s;
read(n);
for(int i=;i<n;++i)
{
read(fa[i+]); read(s); read(P[i+]); read(Q[i+]);
add(fa[i+],i+,s);
}
dfs(,);
for(int i=;i<=n;++i) cout<<mi[i]<<'\n';
}
斜率优化,暴力出队:
#include<cstdio>
#include<iostream> using namespace std; #define N 1000001 typedef long long LL; int front[N],nxt[N],to[N],tot,val[N]; int P[N],Q[N]; int q[N],head,tail; LL dis[N];
LL dp[N]; void read(int &x)
{
x=; char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) { x=x*+c-''; c=getchar(); }
} void add(int u,int v,int w)
{
to[++tot]=v; nxt[tot]=front[u]; front[u]=tot; val[tot]=w;
} inline double X(int i,int j) { return dis[j]-dis[i]; }
inline double Y(int i,int j) { return dp[j]-dp[i]; } void dfs(int x)
{
int now_h=head,now_t=tail;
while(head<tail- && Y(q[head],q[head+])<P[x]*X(q[head],q[head+])) head++;
int j=q[head];
dp[x]=(dis[x]-dis[j])*P[x]+dp[j]+Q[x];
while(head<tail- && Y(q[tail-],q[tail-])*X(q[tail-],x)>X(q[tail-],q[tail-])*Y(q[tail-],x)) tail--;
int rr=q[tail];
q[tail++]=x;
for(int i=front[x];i;i=nxt[i])
dis[to[i]]=dis[x]+val[i],dfs(to[i]);
head=now_h; q[tail-]=rr; tail=now_t;
} int main()
{
int n;
read(n);
int fa,d;
for(int i=;i<=n;++i)
{
read(fa); read(d);
add(fa,i,d);
read(P[i]); read(Q[i]);
}
for(int i=front[];i;i=nxt[i])
{
dis[to[i]]=val[i];
q[head=]=; tail=;
dfs(to[i]);
}
for(int i=;i<=n;++i) cout<<dp[i]<<'\n';
}
斜率优化,二分出队
#include<cstdio>
#include<iostream> using namespace std; #define N 1000001 typedef long long LL; int front[N],nxt[N],to[N],tot,val[N]; int P[N],Q[N]; int q[N],head,tail; LL dis[N];
LL dp[N]; void read(int &x)
{
x=; char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) { x=x*+c-''; c=getchar(); }
} void add(int u,int v,int w)
{
to[++tot]=v; nxt[tot]=front[u]; front[u]=tot; val[tot]=w;
} inline double X(int i,int j) { return dis[j]-dis[i]; }
inline double Y(int i,int j) { return dp[j]-dp[i]; } void dfs(int x)
{
int now_h=head,now_t=tail;
int l=head,r=tail-,mid,tmp=-;
while(l<=r)
{
mid=l+r>>;
if(Y(q[mid],q[mid+])>=P[x]*X(q[mid],q[mid+])) tmp=mid,r=mid-;
else l=mid+;
}
if(tmp!=-) head=tmp;
else head=tail-;
int j=q[head];
dp[x]=(dis[x]-dis[j])*P[x]+dp[j]+Q[x];
l=head,r=tail-,tmp=-;
while(l<=r)
{
mid=l+r>>;
if(Y(q[mid],q[mid+])*X(q[mid+],x)<=X(q[mid],q[mid+])*Y(q[mid+],x)) tmp=mid,l=mid+;
else r=mid-;
}
if(tmp!=-) tail=tmp+;
else tail=head+;
int rr=q[tail];
q[tail++]=x;
for(int i=front[x];i;i=nxt[i])
dis[to[i]]=dis[x]+val[i],dfs(to[i]);
head=now_h; q[tail-]=rr; tail=now_t;
} int main()
{
int n;
read(n);
int fa,d;
for(int i=;i<=n;++i)
{
read(fa); read(d);
add(fa,i,d);
read(P[i]); read(Q[i]);
}
for(int i=front[];i;i=nxt[i])
{
dis[to[i]]=val[i];
q[head=]=; tail=;
dfs(to[i]);
}
for(int i=;i<=n;++i) cout<<dp[i]<<'\n';
}
洛谷 P3994 高速公路的更多相关文章
- 洛谷 P3994 高速公路(斜率优化)
题目链接 题意:给出一棵树,\(1\) 号点为根,边上有边权. 每个点有两个参数 \(p_i,q_i\) 如果你想从 \(i\) 号点到与其距离为 \(d\) 的 \(j\) 号点,那么你需花费 \( ...
- 【洛谷p3994】Highway 二分+斜率优化DP
题目大意:给你一颗$n$个点的有根树,相邻两个点之间有距离,我们可以从$x$乘车到$x$的祖先,费用为$dis\times P[x]+Q[x]$,问你除根以外每个点到根的最小花费. 数据范围:$n≤1 ...
- 洛谷P3994 Highway(树形DP+斜率优化+可持久化线段树/二分)
有点类似NOI2014购票 首先有方程$f(i)=min\{f(j)+(dep_i-dep_j)*p_i+q_i\}$ 这个显然是可以斜率优化的... $\frac {f(j)-f(k)}{dep_j ...
- 洛谷P2221 高速公路【线段树】
题目:https://www.luogu.org/problemnew/show/P2221 题意:有n个节点排成一条链,相邻节点之间有一条路. C u v val表示从u到v的路径上的每条边权值都加 ...
- 洛谷P2242 公路维修问题
To 洛谷.2242 公路维修问题 题目描述 由于长期没有得到维修,A国的高速公路上出现了N个坑.为了尽快填补好这N个坑,A国决定对M处地段采取交通管制.为了求解方便,假设A国的高速公路只有一条,而且 ...
- [洛谷P3761] [TJOI2017]城市
洛谷题目链接:[TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速 ...
- 洛谷 P5638 光骓者的荣耀
洛谷 P5638 [CSGRound2]光骓者的荣耀 洛谷传送门 题目背景 小 K 又在做白日梦了.他进入到他的幻想中,发现他打下了一片江山. 题目描述 小 K 打下的江山一共有nn个城市,城市ii和 ...
- 洛谷 P6383 -『MdOI R2』Resurrection(DP)
洛谷题面传送门 高速公路上正是补 blog 的时候,难道不是吗/doge,难不成逆在高速公路上写题/jy 首先形成的图显然是连通图并且有 \(n-1\) 条边.故形成的图是一棵树. 我们考虑什么样的树 ...
- 洛谷1640 bzoj1854游戏 匈牙利就是又短又快
bzoj炸了,靠离线版题目做了两道(过过样例什么的还是轻松的)但是交不了,正巧洛谷有个"大牛分站",就转回洛谷做题了 水题先行,一道傻逼匈牙利 其实本来的思路是搜索然后发现写出来类 ...
随机推荐
- 在Windows商店应用中使用浅色主题
在开发商店应用时会遇到这样的情况,设计师给我们的设计是浅色背景/深色文本,而商店应用默认是深色背景/浅色文本.那我们需要在每个页面去显式声明背景色和前景色吗,这显然是不理想的.这时就需要设置应用的主题 ...
- 一个Python开源项目-哈勃沙箱源码剖析(下)
前言 在上一篇中,我们讲解了哈勃沙箱的技术点,详细分析了静态检测和动态检测的流程.本篇接着对动态检测的关键技术点进行分析,包括strace,sysdig,volatility.volatility的介 ...
- 微信小程序初体验与DEMO分享
前言 前一段时间微信公布小程序,瞬间引来了大量的关注.博主的公司也将其定为目标之一,遂派本菜为先头兵(踩坑侠). 这次开发了一个比较完整的DEMO,模仿自某个APP首页,由于保护隐私的目的我把数据拷贝 ...
- C#_Winform_聊天机器人
最近研究微信公众平台,搭建了一个微信聊天机器人,调用小黄鸡的公众接口,实现在线和小黄鸡聊天的功能. 接口调用不是很麻烦,不过是php版本,所以研究了一下C#的功能模块, Winfrom版 using ...
- Unity角色对话
对话类------------------------------------------------------------------------------------------------- ...
- python-两个图片相似度算法
# -*- coding: UTF-8 -*- """ 作者:zxj 版本:1.0 日期:19-3-24 """ import cv2 im ...
- 微软职位内部推荐-Sr. SW Engineer for Privacy Id
微软近期Open的职位: Job posting title: Senior Software Engineer for Privacy Identification Profession: Engi ...
- 20135316王剑桥Linux内核学习笔记
王剑桥Linux内核学习笔记 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 计算机是如何工作的 个人理 ...
- 分析code
1 using System; //跟系统说明一下可能会用到这个dll里面的东西 using System.Collections.Generic; //引用集合类命名空间 using System. ...
- Zoom 会议系统
Jfrog的培训过程中 发现ppt的效果很不理想 讲师使用zoom的方式效果很好 首先说一下 zoom的定价体系 官网信息: https://www.zoom.us/profile 好像必须使用 企 ...