Description

 今年夏天,NOI在SZ市迎来了她30周岁的生日。来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会。
       全国的城市构成了一棵以SZ市为根的有根树,每个城市与它的父亲用道路连接。为了方便起见,我们将全国的 n 个城市用 1 到 n 的整数编号。其中SZ市的编号为 1。对于除SZ市之外的任意一个城市 v,我们给出了它在这棵树上的父亲城市 fv  以及到父亲城市道路的长度 sv
从城市 v 前往SZ市的方法为:选择城市 v 的一个祖先 a,支付购票的费用,乘坐交通工具到达 a。再选择城市 a 的一个祖先 b,支付费用并到达 b。以此类推,直至到达SZ市。
对于任意一个城市 v,我们会给出一个交通工具的距离限制 lv。对于城市 v 的祖先 a,只有当它们之间所有道路的总长度不超过 lv  时,从城市 v 才可以通过一次购票到达城市 a,否则不能通过一次购票到达。对于每个城市 v,我们还会给出两个非负整数 pv,qv  作为票价参数。若城市 v 到城市 a 所有道路的总长度为 d,那么从城市 v 到城市 a 购买的票价为 dpv+qv
每个城市的OIer都希望自己到达SZ市时,用于购票的总资金最少。你的任务就是,告诉每个城市的OIer他们所花的最少资金是多少。

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

7 3
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

40
150
70
149
300
150

解题思路:

这道题的状态转移方程非常好列,Dp[i]=min(Dp[anc[i]]+p*disi,anc[i]+q)

这个可以斜率优化我就不说了。

像序列上的CDQ,先处理左半部分更新右半部分。

主要是先处理i到根的所有节点Dp值来更新重心i,再将更深的子树内按照失效大小排序,就可以不断地实现加点单调栈维护凸包。

注意加根反着加,所以要将x轴反过来(当然你递归处理的话就用不着了)

注意inf要足够大。

注意要动态更新答案,防止优秀点失效。

代码:

  1. #include<cstdio>
  2. #include<cstring>
  3. #include<algorithm>
  4. typedef long long lnt;
  5. const int N=;
  6. const double eps=1e-;
  7. struct pnt{
  8. int no;
  9. int hd;
  10. int fa;
  11. int wgt;
  12. lnt f,dis,p,q,l;
  13. bool vis;
  14. double x(void){return dis;}
  15. double y(void){return f;}
  16. double k(void){return p;}
  17. }p[N];
  18. struct ent{
  19. int twd;
  20. int lst;
  21. lnt vls;
  22. }e[N<<];
  23. int n,m;
  24. int cnt;
  25. int toa;
  26. int tob;
  27. int top;
  28. int root;
  29. int size;
  30. int maxsize;
  31. int sta[N];
  32. int stb[N];
  33. int stack[N];
  34. bool cmp(int a,int b)
  35. {
  36. return p[a].dis-p[a].l>p[b].dis-p[b].l;
  37. }
  38. double K(int a,int b)
  39. {
  40. return (double)(p[a].y()-p[b].y())/(double)(p[a].x()-p[b].x());
  41. }
  42. void ade(int f,int t,lnt v)
  43. {
  44. cnt++;
  45. e[cnt].twd=t;
  46. e[cnt].lst=p[f].hd;
  47. e[cnt].vls=v;
  48. p[f].hd=cnt;
  49. return ;
  50. }
  51. void grc_dfs(int x,int f)
  52. {
  53. p[x].wgt=;
  54. int maxs=-;
  55. for(int i=p[x].hd;i;i=e[i].lst)
  56. {
  57. int to=e[i].twd;
  58. if(to==f||p[to].vis)
  59. continue;
  60. grc_dfs(to,x);
  61. p[x].wgt+=p[to].wgt;
  62. if(maxs<p[to].wgt)
  63. maxs=p[to].wgt;
  64. }
  65. if(maxs<size-p[x].wgt)
  66. maxs=size-p[x].wgt;
  67. if(maxs<maxsize)
  68. {
  69. root=x;
  70. maxsize=maxs;
  71. }
  72. return ;
  73. }
  74. void get_ans(int x)
  75. {
  76. if(!top)
  77. return ;
  78. int l=,r=top-;
  79. int y=stack[top];
  80. while(l<=r)
  81. {
  82. int mid=(l+r)>>;
  83. if(K(stack[mid],stack[mid+])<p[x].k())
  84. r=mid-,y=stack[mid];
  85. else
  86. l=mid+;
  87. }
  88. p[x].f=std::min(p[x].f,p[y].f+(p[x].dis-p[y].dis)*p[x].p+p[x].q);
  89. return ;
  90. }
  91. void Insert(int x,int f)
  92. {
  93. stb[++tob]=x;
  94. for(int i=p[x].hd;i;i=e[i].lst)
  95. {
  96. int to=e[i].twd;
  97. if(to==f||p[to].vis)
  98. continue;
  99. Insert(to,x);
  100. }
  101. return ;
  102. }
  103. void CDQ(int x)
  104. {
  105. int rt;
  106. root=;
  107. size=p[x].wgt;
  108. maxsize=0x3f3f3f3f;
  109. grc_dfs(x,x);
  110. rt=root;
  111. p[rt].vis=true;
  112. if(rt!=x)
  113. {
  114. p[x].wgt-=p[rt].wgt;
  115. CDQ(x);
  116. }
  117. toa=tob=top=;
  118. sta[++toa]=rt;
  119. for(int i=rt;i!=x;i=p[i].fa)
  120. {
  121. if(p[rt].dis-p[p[i].fa].dis<=p[rt].l)
  122. 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);
  123. sta[++toa]=p[i].fa;
  124. }
  125. for(int i=p[rt].hd;i;i=e[i].lst)
  126. {
  127. int to=e[i].twd;
  128. if(p[to].vis)
  129. continue;
  130. Insert(to,to);
  131. }
  132. std::sort(stb+,stb+tob+,cmp);
  133. int j=;
  134. for(int i=;i<=toa;i++)
  135. {
  136. while(j<=tob&&p[stb[j]].dis-p[sta[i]].dis>p[stb[j]].l)
  137. get_ans(stb[j++]);
  138. while(top>&&K(stack[top-],stack[top])<=K(stack[top],sta[i]))
  139. top--;
  140. stack[++top]=sta[i];
  141. }
  142. while(j<=tob)
  143. get_ans(stb[j++]);
  144. for(int i=p[rt].hd;i;i=e[i].lst)
  145. {
  146. int to=e[i].twd;
  147. if(p[to].vis)
  148. continue;
  149. CDQ(to);
  150. }
  151. return ;
  152. }
  153. void dis_measure(int x,int f)
  154. {
  155. for(int i=p[x].hd;i;i=e[i].lst)
  156. {
  157. int to=e[i].twd;
  158. if(to==f)
  159. continue;
  160. p[to].dis=p[x].dis+e[i].vls;
  161. dis_measure(to,x);
  162. }
  163. return ;
  164. }
  165. int main()
  166. {
  167. p[].f=0x3f3f3f3f3f3f3f3fll;
  168. scanf("%d%d",&n,&m);
  169. for(int i=;i<=n;i++)
  170. {
  171. p[i].no=i;
  172. p[i].f=0x3f3f3f3f3f3f3f3fll;
  173. lnt tmp;
  174. scanf("%d%lld%lld%lld%lld",&p[i].fa,&tmp,&p[i].p,&p[i].q,&p[i].l);
  175. ade(i,p[i].fa,tmp);
  176. ade(p[i].fa,i,tmp);
  177. }
  178. dis_measure(,);
  179. p[].fa=;
  180. p[].wgt=n;
  181. CDQ();
  182. for(int i=;i<=n;i++)
  183. printf("%lld\n",p[i].f);
  184. return ;
  185. }
 

BZOJ3672: [Noi2014]购票(CDQ分治,点分治)的更多相关文章

  1. [BZOJ3672][Noi2014]购票 斜率优化+点分治+cdq分治

    3672: [Noi2014]购票 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 1749  Solved: 885[Submit][Status][ ...

  2. bzoj千题计划251:bzoj3672: [Noi2014]购票

    http://www.lydsy.com/JudgeOnline/problem.php?id=3672 法一:线段树维护可持久化单调队列维护凸包 斜率优化DP 设dp[i] 表示i号点到根节点的最少 ...

  3. BZOJ3672: [Noi2014]购票【CDQ分治】【点分治】【斜率优化DP】

    Description 今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会. 全国的城市构成了一棵以SZ市为根的有根树,每个城市与它的 ...

  4. BZOJ3672 [Noi2014]购票 【点分治 + 斜率优化】

    题目链接 BZOJ3672 题解 如果暂时不管\(l[i]\)的限制,并假使这是一条链 设\(f[i]\)表示\(i\)节点的最优答案,我们容易得到\(dp\)方程 \[f[i] = min\{f[j ...

  5. BZOJ3672: [Noi2014]购票(dp 斜率优化 点分治 二分 凸包)

    题意 题目链接 Sol 介绍一种神奇的点分治的做法 啥?这都有根树了怎么点分治?? 嘿嘿,这道题的点分治不同于一般的点分治.正常的点分治思路大概是先统计过重心的,再递归下去 实际上一般的点分治与统计顺 ...

  6. 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]) 对这棵树进行点分治,每次递 ...

  7. bzoj3672: [Noi2014]购票(树形DP+斜率优化+可持久化凸包)

    这题的加强版,多了一个$l_i$的限制,少了一个$p_i$的单调性,难了好多... 首先有方程$f(i)=min\{f(j)+(dep_i-dep_j)*p_i+q_i\}$ $\frac {f(j) ...

  8. 【BZOJ3672】[Noi2014]购票 树分治+斜率优化

    [BZOJ3672][Noi2014]购票 Description  今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会.       ...

  9. 【BZOJ 3672】 3672: [Noi2014]购票 (CDQ分治+点分治+斜率优化)**

    3672: [Noi2014]购票 Description  今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会.        全国 ...

随机推荐

  1. Pycharm 的安装

    一. Windows 安装 汉化 破解补丁激活 下载 `https://pan.baidu.com/s/1qjI9uHaw0x374rwu6H8djA` 并将 JetbrainsCrack-2.8-r ...

  2. org.mybatis.spring.mapper.MapperScannerConfigurer$Scanner$1

    不能加载或找不到 org.mybatis.spring.mapper.MapperScannerConfigurer$Scanner$1 经查证,是mybatis-spring-xxx.jar 这个版 ...

  3. IPv4私有IP地址有哪些!

    私有IP地址是一段保留的IP地址.只是使用在局域网中,在Internet上是不使用的. 私有IP地址的范围有: 私网地址分有三类, A类中,第一段为10的都为私网地址,B类中,以172.16--172 ...

  4. 紫书 习题 10-21 UVa 1649 (组合数)

    C(n, k) = m, 固定k,枚举k 这里用到了组合数的一个性质 当k固定的时候,C(2 * k, k) 最小 C(m, k)最大(对于这道题而言是这样,因为大于m 就最终答案不可能为m了) 所以 ...

  5. QT项目升级(QT4.6.3到QT5.2)时,遇到的问题和解决方法

    QT4升级到QT5修改: PC部分: [改QTDIR变量] 在project根文件夹下找到.user文件, 如InnoTabPlugin.vcxproj.user 改动指向你的QT5根文件夹: < ...

  6. jquery11源码 animate() : 运动的方法

    { var fxNow, timerId, rfxtypes = /^(?:toggle|show|hide)$/, rfxnum = new RegExp( "^(?:([+-])=|)( ...

  7. elasticsearch index 之 create index(-)

    从本篇开始,就进入了Index的核心代码部分.这里首先分析一下索引的创建过程.elasticsearch中的索引是多个分片的集合,它只是逻辑上的索引,并不具备实际的索引功能,所有对数据的操作最终还是由 ...

  8. Js将类数组转化为数组

    说起伪数组,大家可能会想到arguments, 这个我们函数参数的一个类数组,是类数组的代表. 1.拥有length属性,可以使用下标来访问元素,这两点和数组相同. 2.不能使用数组的方法,他们不能使 ...

  9. assert增强宏的实现

    作者:朱金灿 来源:http://blog.csdn.net/clever101 标准c的assert宏和MFC的ASSERT宏都不支持输出太多的信息.今天实现了一个assert增强宏,可以输出更多的 ...

  10. 浏览器下管理Linux系统--记webmin的使用

    本文介绍一款浏览器方式来管理linux的一种方式,这款软件就叫webmin,Webmin 让您能够在远程使用支持 HTTPS (SSL 上的 HTTP)协议的 Web 浏览器通过 Web 界面管理您的 ...