bzoj3672【NOI2014】购票
题目描述
输入格式
第 1 行包含2个非负整数 n,t,分别表示城市的个数和数据类型(其意义将在后面提到)。输入文件的第 2 到 n 行,每行描述一个除SZ之外的城市。其中第 v 行包含 5 个非负整数 $f_v,s_v,p_v,q_v,l_v$,分别表示城市 v 的父亲城市,它到父亲城市道路的长度,票价的两个参数和距离限制。请注意:输入不包含编号为 1 的SZ市,第 2 行到第 n 行分别描述的是城市 2 到城市 n。
输出格式
输出包含 n-1 行,每行包含一个整数。其中第 v 行表示从城市 v+1 出发,到达SZ市最少的购票费用。同样请注意:输出不包含编号为 1 的SZ市。
数据规模
题解:
- 设$v$是$u$的祖先,$dp[u] = dp[v] + (dis[u]-dis[v])*p[u] + q[u] \ , \ (dis[u]-dis[v]<=l[u])$
- 斜率优化部分:
- 假设$d[v_{1}]<d[v_{2}]$且对$u$来说$v2$比$v1$更优:
- $$dp[v_{1}]+(dis[u]-dis[v_{1}])*p[u]+q[u] > dp[v_{2}]+(dis[u]-dis[v_{2}])*p[u]+q[u] ;$$
- $$\frac{dp[v_{2}]-dp[v_{1}]}{dis[v2]-dis[v1]} < p[u]$$
- 可以维护$(dis[v],dp[v])$的下凸包(斜率单调上升);
- 由于是在一颗树上且有$l$的限制,所以我们需要用用一些方法维护一下,
- 可以找树的重心分治,我写的树剖:
- 对每个重链的顶端维护一个凸包,$dfs$的时候不断加入点;
- 我们先$dfs$轻儿子,这样一个点到根的轻链的父亲都是一个凸包的末端;
- 找转移点的时候不断向上跳,如果某条链全部满足限制则在重链顶端的凸包上直接查询;
- 否则一定是合法的分界点在当前找到的链上,用线段树维护区间的凸包查询即可;
- 一次最多找到$log$条树链,最多有一条树链需要在线段树里面查询;
- 算上三分,时间复杂度:$O(nlog^2n)$
- 有个小技巧,线段树满了之后再做凸包
#include<bits/stdc++.h>
#define ls (k<<1)
#define rs (k<<1|1)
#define ll long long
#define ld double
using namespace std;
const int N=;
char gc(){
static char*p1,*p2,s[];
if(p1==p2)p2=(p1=s)+fread(s,,,stdin);
return(p1==p2)?EOF:*p1++;
}
ll rd(){
ll x=; char c=gc();
while(c<''||c>'')c=gc();
while(c>=''&&c<='')x=(x<<)+(x<<)+c-'',c=gc();
return x;
}
int n,T,f[N],o=,hd[N],dep[N],fa[N][],bin[];
int idx,sz[N],tp[N],sn[N],mx[N],id[N],cnt[N<<],st[N];
ll dp[N],tmp,dis[N],s[N],p[N],q[N],l[N];
struct Poi{
ld x,y;
Poi(ld _x=,ld _y=):x(_x),y(_y){};
Poi operator -(const Poi&a){return Poi(x-a.x,y-a.y);}
ld operator ^(const Poi&a){return x*a.y-y*a.x;}
bool operator <(const Poi&a){return x==a.x?y<a.y:x<a.x;}
}P[N];
vector<int>g1[N<<],g2[N];
struct Edge{int v,nt;}E[N];
void adde(int u,int v){E[o]=(Edge){v,hd[u]};hd[u]=o++;}
void dfs1(int u){
sz[u]=;
fa[u][]=f[u];
dep[u]=dep[f[u]]+;
dis[u]=dis[f[u]]+s[u];
for(int i=;bin[i]<dep[u];++i)fa[u][i]=fa[fa[u][i-]][i-];
for(int i=hd[u];i;i=E[i].nt){
int v=E[i].v;
dfs1(v);
sz[u]+=sz[v];
if(sz[v]>sz[sn[u]])sn[u]=v;
}
}
void dfs2(int u,int t){
tp[u]=t;
id[u]=++idx;
st[idx]=u;
if(sn[u])dfs2(sn[u],t);
for(int i=hd[u];i;i=E[i].nt){
int v=E[i].v;
if(v==sn[u])continue;
dfs2(v,v);
}
}
inline bool judge(int a,int b,int c){return ((P[b]-P[a])^(P[c]-P[b]))<=;}
void pushup(int k){
int l=ls,r=rs,tl=,tr=;
while(tl<(int)g1[l].size()||tr<(int)g1[r].size()){
if(tr==(int)g1[r].size()||(tl<(int)g1[l].size()&&g1[l][tl]<g1[r][tr])){
while(g1[k].size()>&&judge(g1[k][g1[k].size()-],g1[k].back(),g1[l][tl]))g1[k].pop_back();
g1[k].push_back(g1[l][tl++]);
}else{
while(g1[k].size()>&&judge(g1[k][g1[k].size()-],g1[k].back(),g1[r][tr]))g1[k].pop_back();
g1[k].push_back(g1[r][tr++]);
}
}
}
void ins1(int k,int l,int r,int u){
cnt[k]++;
if(l==r){g1[k].push_back(u);return;}
int mid=(l+r)>>;
if(id[u]<=mid)ins1(ls,l,mid,u);
else ins1(rs,mid+,r,u);
if(tp[st[l]]==tp[st[r]]&&cnt[k]==r-l+)pushup(k);
}
void ins2(int k,int u){
while(g2[k].size()>&&judge(g2[k][g2[k].size()-],g2[k].back(),u))g2[k].pop_back();
g2[k].push_back(u);
}
inline bool ok(int u,int v){return dis[u]-dis[v]<=l[u];}
inline void upd(ll&x,ll y){if(x>y)x=y;}
void cal(vector<int>&g,int u){
int l=,r=g.size()-;
while(r-l>){
int mid=(r-l)/,mid1=l+mid,mid2=r-mid;
ll t1=dp[g[mid1]]-dis[g[mid1]]*p[u];
ll t2=dp[g[mid2]]-dis[g[mid2]]*p[u];
if(t1>t2)l=mid1+;
else r=mid2-;
}
for(int i=l;i<=r;++i)upd(tmp,dp[g[i]]-dis[g[i]]*p[u]);
}
void query1(int k,int l,int r,int x,int y,int u){
if(l==x&&r==y){cal(g1[k],u);return;}
int mid=(l+r)>>;
if(y<=mid)query1(ls,l,mid,x,y,u);
else if(x>mid)query1(rs,mid+,r,x,y,u);
else query1(ls,l,mid,x,mid,u),query1(rs,mid+,r,mid+,y,u);
}
void query2(int x,int u){
cal(g2[x],u);
}
void solve(int u){
tmp=u==?:9e18;
for(int x=f[u],t=tp[x];x;x=f[t],t=tp[x]){
if(dep[mx[u]]<dep[t])query2(id[t],u);
else {query1(,,n,id[mx[u]],id[x],u);break;}
}
dp[u]=dis[u]*p[u]+q[u]+tmp;
P[u]=Poi(dis[u],dp[u]);
ins1(,,n,u);
ins2(id[tp[u]],u);
}
void calmx(int u){
int t=u;
for(int i=;~i;--i)if(fa[t][i]&&ok(u,fa[t][i]))t=fa[t][i];
mx[u]=t;
}
void dfs3(int u){
calmx(u);
solve(u);
for(int i=hd[u];i;i=E[i].nt){
int v=E[i].v;
if(v==sn[u])continue;
dfs3(v);
}
if(sn[u])dfs3(sn[u]);
}
int main(){
#ifndef ONLINE_JUDGE
freopen("bzoj3672.in","r",stdin);
freopen("bzoj3672.out","w",stdout);
#endif
for(int i=bin[]=;i<=;++i)bin[i]=bin[i-]<<;
n=rd();T=rd();
for(int i=;i<=n;++i){
adde(f[i]=rd(),i);
s[i]=rd();p[i]=rd();
q[i]=rd();l[i]=rd();
}
dfs1();
dfs2(,);
dfs3();
for(int i=;i<=n;++i)printf("%lld\n",dp[i]);
return ;
}
bzoj3672【NOI2014】购票的更多相关文章
- bzoj千题计划251:bzoj3672: [Noi2014]购票
http://www.lydsy.com/JudgeOnline/problem.php?id=3672 法一:线段树维护可持久化单调队列维护凸包 斜率优化DP 设dp[i] 表示i号点到根节点的最少 ...
- [BZOJ3672][Noi2014]购票 斜率优化+点分治+cdq分治
3672: [Noi2014]购票 Time Limit: 30 Sec Memory Limit: 512 MBSubmit: 1749 Solved: 885[Submit][Status][ ...
- BZOJ3672: [Noi2014]购票【CDQ分治】【点分治】【斜率优化DP】
Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. 全国的城市构成了一棵以SZ市为根的有根树,每个城市与它的 ...
- BZOJ3672 [Noi2014]购票 【点分治 + 斜率优化】
题目链接 BZOJ3672 题解 如果暂时不管\(l[i]\)的限制,并假使这是一条链 设\(f[i]\)表示\(i\)节点的最优答案,我们容易得到\(dp\)方程 \[f[i] = min\{f[j ...
- BZOJ3672: [Noi2014]购票(CDQ分治,点分治)
Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. 全国的城市构成了一棵以SZ市为根的有根树 ...
- BZOJ3672 : [Noi2014]购票
设d[i]表示i到1的距离 f[i]=w[i]+min(f[j]+(d[i]-d[j])*v[i])=w[i]+d[i]*v[i]+min(-d[j]*v[i]+f[j]) 对这棵树进行点分治,每次递 ...
- BZOJ3672: [Noi2014]购票(dp 斜率优化 点分治 二分 凸包)
题意 题目链接 Sol 介绍一种神奇的点分治的做法 啥?这都有根树了怎么点分治?? 嘿嘿,这道题的点分治不同于一般的点分治.正常的点分治思路大概是先统计过重心的,再递归下去 实际上一般的点分治与统计顺 ...
- bzoj3672: [Noi2014]购票(树形DP+斜率优化+可持久化凸包)
这题的加强版,多了一个$l_i$的限制,少了一个$p_i$的单调性,难了好多... 首先有方程$f(i)=min\{f(j)+(dep_i-dep_j)*p_i+q_i\}$ $\frac {f(j) ...
- [BZOJ3672][UOJ#7][NOI2014]购票
[BZOJ3672][UOJ#7][NOI2014]购票 试题描述 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. ...
- 【BZOJ3672】[Noi2014]购票 树分治+斜率优化
[BZOJ3672][Noi2014]购票 Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. ...
随机推荐
- 基本数据结构 -- 栈简介(C语言实现)
栈是一种后进先出的线性表,是最基本的一种数据结构,在许多地方都有应用. 一.什么是栈 栈是限制插入和删除只能在一个位置上进行的线性表.其中,允许插入和删除的一端位于表的末端,叫做栈顶(top),不允许 ...
- [转载]java面试中经常会被问到的一些算法的问题
Java面试中经常会被问到的一些算法的问题,而大部分算法的理论及思想,我们曾经都能倒背如流,并且也能用开发语言来实现过, 可是很多由于可能在项目开发中应用的比较少,久而久之就很容易被忘记了,在此我分享 ...
- SmartRaiden 和 Lighting Network 进行去中心化跨链原子资产交换
作者介绍 虫洞社区·签约作者 steven bai 前言 如果能够进行以太坊和比特币跨链原子资产交换,是不是一件很酷的事情? 目前链下的扩容方式有很多,最广为人知的就是比特币的闪电网络和以太坊的雷电网 ...
- whoami,who,w命令详解
http://www.voidcn.com/blog/wszzdanm/article/p-6145895.html 命令功能:显示登录用户的信息 命令格式: 常用选项: 举例: w 显示已经登录的用 ...
- yum安装lnmp
python其他知识目录 1.安装LNMP之前要安装EPEL,以便安装源以外的软件,如Nginx,phpMyAdmin等. yum install epel-release 提示:EPEL,即Extr ...
- [2019BUAA软工]第0次代码作业
Visual Studio 单元测试的简单应用 写在前面 笔者根据作业的介绍以及Visual Studio 2017 文档的相关说明,进行了Visual Studio 单元测试的简单应用. Tip ...
- vim搭建C编程IDE
曾经在一篇关于vim技巧的文章里有一句话:"世界上只有三种编辑器,EMACS.VIM和其它." 我不知道这是不是太过于绝对了,但是从我所看到的每一篇linux下编程以及文字编辑的文 ...
- 20172325 2017-2018-2 《Java程序设计》第五周学习总结
20172325 2017-2018-2 <Java程序设计>第五周学习总结 教材学习内容总结 1.布尔表达式的值只有真或假,表达式的结果决定了下一步将要执行的语句. 2.循环语句可以用在 ...
- 今日事——Sprint计划会议
一. Sprint需求: 解屏提醒部分 界面设计 登录功能 备忘功能 成就系统 二.工作认领: 因有成员请假回家,所以延后认领,目前主要任务是学习如何在andriod平台开发并搭建开发环境. 网上 ...
- Head First Java & final