BZOJ3672: [Noi2014]购票(CDQ分治,点分治)
Description
Input
第 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。
Output
输出包含 n-1 行,每行包含一个整数。其中第 v 行表示从城市 v+1 出发,到达SZ市最少的购票费用。同样请注意:输出不包含编号为 1 的SZ市。
Sample Input
1 2 20 0 3
1 5 10 100 5
2 4 10 10 10
2 9 1 100 10
3 5 20 100 10
4 4 20 0 10
Sample Output
150
70
149
300
150
解题思路:
这道题的状态转移方程非常好列,Dp[i]=min(Dp[anc[i]]+p*disi,anc[i]+q)
这个可以斜率优化我就不说了。
像序列上的CDQ,先处理左半部分更新右半部分。
主要是先处理i到根的所有节点Dp值来更新重心i,再将更深的子树内按照失效大小排序,就可以不断地实现加点单调栈维护凸包。
注意加根反着加,所以要将x轴反过来(当然你递归处理的话就用不着了)
注意inf要足够大。
注意要动态更新答案,防止优秀点失效。
代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
typedef long long lnt;
const int N=;
const double eps=1e-;
struct pnt{
int no;
int hd;
int fa;
int wgt;
lnt f,dis,p,q,l;
bool vis;
double x(void){return dis;}
double y(void){return f;}
double k(void){return p;}
}p[N];
struct ent{
int twd;
int lst;
lnt vls;
}e[N<<];
int n,m;
int cnt;
int toa;
int tob;
int top;
int root;
int size;
int maxsize;
int sta[N];
int stb[N];
int stack[N];
bool cmp(int a,int b)
{
return p[a].dis-p[a].l>p[b].dis-p[b].l;
}
double K(int a,int b)
{
return (double)(p[a].y()-p[b].y())/(double)(p[a].x()-p[b].x());
}
void ade(int f,int t,lnt v)
{
cnt++;
e[cnt].twd=t;
e[cnt].lst=p[f].hd;
e[cnt].vls=v;
p[f].hd=cnt;
return ;
}
void grc_dfs(int x,int f)
{
p[x].wgt=;
int maxs=-;
for(int i=p[x].hd;i;i=e[i].lst)
{
int to=e[i].twd;
if(to==f||p[to].vis)
continue;
grc_dfs(to,x);
p[x].wgt+=p[to].wgt;
if(maxs<p[to].wgt)
maxs=p[to].wgt;
}
if(maxs<size-p[x].wgt)
maxs=size-p[x].wgt;
if(maxs<maxsize)
{
root=x;
maxsize=maxs;
}
return ;
}
void get_ans(int x)
{
if(!top)
return ;
int l=,r=top-;
int y=stack[top];
while(l<=r)
{
int mid=(l+r)>>;
if(K(stack[mid],stack[mid+])<p[x].k())
r=mid-,y=stack[mid];
else
l=mid+;
}
p[x].f=std::min(p[x].f,p[y].f+(p[x].dis-p[y].dis)*p[x].p+p[x].q);
return ;
}
void Insert(int x,int f)
{
stb[++tob]=x;
for(int i=p[x].hd;i;i=e[i].lst)
{
int to=e[i].twd;
if(to==f||p[to].vis)
continue;
Insert(to,x);
}
return ;
}
void CDQ(int x)
{
int rt;
root=;
size=p[x].wgt;
maxsize=0x3f3f3f3f;
grc_dfs(x,x);
rt=root;
p[rt].vis=true;
if(rt!=x)
{
p[x].wgt-=p[rt].wgt;
CDQ(x);
}
toa=tob=top=;
sta[++toa]=rt;
for(int i=rt;i!=x;i=p[i].fa)
{
if(p[rt].dis-p[p[i].fa].dis<=p[rt].l)
p[rt].f=std::min(p[rt].f,p[p[i].fa].f+(p[rt].dis-p[p[i].fa].dis)*p[rt].p+p[rt].q);
sta[++toa]=p[i].fa;
}
for(int i=p[rt].hd;i;i=e[i].lst)
{
int to=e[i].twd;
if(p[to].vis)
continue;
Insert(to,to);
}
std::sort(stb+,stb+tob+,cmp);
int j=;
for(int i=;i<=toa;i++)
{
while(j<=tob&&p[stb[j]].dis-p[sta[i]].dis>p[stb[j]].l)
get_ans(stb[j++]);
while(top>&&K(stack[top-],stack[top])<=K(stack[top],sta[i]))
top--;
stack[++top]=sta[i];
}
while(j<=tob)
get_ans(stb[j++]);
for(int i=p[rt].hd;i;i=e[i].lst)
{
int to=e[i].twd;
if(p[to].vis)
continue;
CDQ(to);
}
return ;
}
void dis_measure(int x,int f)
{
for(int i=p[x].hd;i;i=e[i].lst)
{
int to=e[i].twd;
if(to==f)
continue;
p[to].dis=p[x].dis+e[i].vls;
dis_measure(to,x);
}
return ;
}
int main()
{
p[].f=0x3f3f3f3f3f3f3f3fll;
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
{
p[i].no=i;
p[i].f=0x3f3f3f3f3f3f3f3fll;
lnt tmp;
scanf("%d%lld%lld%lld%lld",&p[i].fa,&tmp,&p[i].p,&p[i].q,&p[i].l);
ade(i,p[i].fa,tmp);
ade(p[i].fa,i,tmp);
}
dis_measure(,);
p[].fa=;
p[].wgt=n;
CDQ();
for(int i=;i<=n;i++)
printf("%lld\n",p[i].f);
return ;
}
BZOJ3672: [Noi2014]购票(CDQ分治,点分治)的更多相关文章
- [BZOJ3672][Noi2014]购票 斜率优化+点分治+cdq分治
3672: [Noi2014]购票 Time Limit: 30 Sec Memory Limit: 512 MBSubmit: 1749 Solved: 885[Submit][Status][ ...
- bzoj千题计划251:bzoj3672: [Noi2014]购票
http://www.lydsy.com/JudgeOnline/problem.php?id=3672 法一:线段树维护可持久化单调队列维护凸包 斜率优化DP 设dp[i] 表示i号点到根节点的最少 ...
- 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]购票(dp 斜率优化 点分治 二分 凸包)
题意 题目链接 Sol 介绍一种神奇的点分治的做法 啥?这都有根树了怎么点分治?? 嘿嘿,这道题的点分治不同于一般的点分治.正常的点分治思路大概是先统计过重心的,再递归下去 实际上一般的点分治与统计顺 ...
- 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+斜率优化+可持久化凸包)
这题的加强版,多了一个$l_i$的限制,少了一个$p_i$的单调性,难了好多... 首先有方程$f(i)=min\{f(j)+(dep_i-dep_j)*p_i+q_i\}$ $\frac {f(j) ...
- 【BZOJ3672】[Noi2014]购票 树分治+斜率优化
[BZOJ3672][Noi2014]购票 Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. ...
- 【BZOJ 3672】 3672: [Noi2014]购票 (CDQ分治+点分治+斜率优化)**
3672: [Noi2014]购票 Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. 全国 ...
随机推荐
- nginx 11个过程
nginx在处理每一个用户请求时,都是按照若干个不同的阶段依次处理的,与配置文件上的顺序没有关系,详细内容可以阅读<深入理解nginx:模块开发与架构解析>这本书,这里只做简单介绍: 1. ...
- django 笔记3
FBV function base view url.py index -> 函数名 view.py def 函数(requset): ... CBV class base view /inde ...
- phpMyAdmin出现找不到mysql扩展和Cannot log in to the MySQL server问题
环境:Centos6.5,Apache2.4, PHP5.5, MySql5.6. phpMyAdmin版本:https://files.phpmyadmin.net/phpMyAdmin/4.4.1 ...
- Gym - 100625F Count Ways 快速幂+容斥原理
题意:n*m的格子,中间有若干点不能走,问从左上角到右下角有多少种走法. 思路:CountWay(i,j) 表示从 i 点到 j 点的种数.然后用容斥原理加加减减解决 #pragma comment( ...
- AnkhSvn介绍 插件
转载:http://www.cnblogs.com/lyhabc/articles/2483011.html AnkhSVN是一款在VS中管理Subversion的插件,您可以在VS中轻松的提交.更新 ...
- c# 引用ConfigurationManager 类
c#添加了Configuration;后,竟然找不到 ConfigurationManager 这个类,后来才发现:虽然引用了using System.Configuration;这个包,但是还是不行 ...
- thinkserer TD350 系统损坏后,数据恢复及系统重做过程
电脑配置: 联想服务器 TD350 E5-2609V4 2*8G 2*4T+R1 塔式 单电 1.系统恢复: 试过很多种方法,均无效 2.数据恢复: 重新安装系统后,直接在D盘查找 , 原C盘的 ...
- Python正则表达式初识(一)
首先跟大家简单唠叨两句为什么要学习正则表达式,为什么在网络爬虫的时候离不开正则表达式.正则表达式在处理字符串的时候扮演着非常重要的角色,在网络爬虫的时候也十分常用,大家可以把它学的简单一些,但是不能不 ...
- jQuery判断字符串是否含有中文字符
//判断字符串是不是中文String.prototype.isChinese = function () { var reg = /[^\x00-\xff]/ig;//判断是否存在中文和全角字符 ...
- MySQL事务(event scheduler)的学习【事务创建之后,没有运行的问题】
[本篇文章主要解决的是,MySQL事务创建之后,没有运行的问题] 首先从这里开始:http://www.w3schools.in/mysql/event-schedule/,创建了基本的MySQL事务 ...