BZOJ2500: 幸福的道路

题解:

一道不错的题目。

树DP可以求出从每个点出发的最长链,复杂度O(n)

然后就变成找一个数列里最长的连续区间使得最大值-最小值<=m了。

成了这题:http://www.cnblogs.com/zyfzyf/p/4008295.html

代码:

  1 #include<cstdio>
  2
  3 #include<cstdlib>
  4
  5 #include<cmath>
  6
  7 #include<cstring>
  8
  9 #include<algorithm>
 10
 11 #include<iostream>
 12
 13 #include<vector>
 14
 15 #include<map>
 16
 17 #include<set>
 18
 19 #include<queue>
 20
 21 #include<string>
 22
 23 #define inf 1000000000
 24
 25 #define maxn 1000000+5
 26
 27 #define maxm 20000000+5
 28
 29 #define eps 1e-10
 30
 31 #define ll long long
 32
 33 #define pa pair<int,int>
 34
 35 #define for0(i,n) for(int i=0;i<=(n);i++)
 36
 37 #define for1(i,n) for(int i=1;i<=(n);i++)
 38
 39 #define for2(i,x,y) for(int i=(x);i<=(y);i++)
 40
 41 #define for3(i,x,y) for(int i=(x);i>=(y);i--)
 42 #define for4(i,x) for(int i=head[x],y;i;i=e[i].next)
 43
 44 #define mod 1000000007
 45
 46 using namespace std;
 47
 48 inline int read()
 49
 50 {
 51
 52     int x=0,f=1;char ch=getchar();
 53
 54     while(ch<‘0‘||ch>‘9‘){if(ch==‘-‘)f=-1;ch=getchar();}
 55
 56     while(ch>=‘0‘&&ch<=‘9‘){x=10*x+ch-‘0‘;ch=getchar();}
 57
 58     return x*f;
 59
 60 }
 61 struct edge{int go,next;ll w;}e[2*maxn];
 62 int n,tot,q[maxn][2],l[2],r[2],head[maxn];
 63 ll m,f[maxn],g[maxn][2],a[maxn];
 64 inline void insert(int x,int y,ll z)
 65 {
 66     e[++tot]=(edge){y,head[x],z};head[x]=tot;
 67 }
 68 inline void down(int x)
 69 {
 70     for4(i,x)
 71     {
 72         down(y=e[i].go);
 73         if(g[y][0]+e[i].w>g[x][0])g[x][1]=g[x][0],g[x][0]=g[y][0]+e[i].w;
 74         else g[x][1]=max(g[x][1],g[y][0]+e[i].w);
 75     }
 76 }
 77 inline void up(int x)
 78 {
 79     for4(i,x)
 80     {
 81         f[y=e[i].go]=f[x]+e[i].w;
 82         if(g[y][0]+e[i].w==g[x][0])f[y]=max(f[y],g[x][1]+e[i].w);
 83         else f[y]=max(f[y],g[x][0]+e[i].w);
 84         up(y);
 85     }
 86 }
 87
 88 int main()
 89
 90 {
 91
 92     freopen("input.txt","r",stdin);
 93
 94     freopen("output.txt","w",stdout);
 95
 96     n=read();m=read();
 97     for2(i,2,n){int x=read(),y=read();insert(x,i,y);}
 98     down(1);up(1);
 99     for1(i,n)a[i]=max(f[i],g[i][0]);
100     l[0]=l[1]=1;r[0]=r[1]=0;
101     int ret=1,ans=0;
102     for1(i,n)
103     {
104        while(l[0]<=r[0]&&a[i]<=a[q[r[0]][0]])r[0]--;
105        q[++r[0]][0]=i;
106        while(l[1]<=r[1]&&a[i]>=a[q[r[1]][1]])r[1]--;
107        q[++r[1]][1]=i;
108        while(a[q[l[1]][1]]-a[q[l[0]][0]]>m)
109            ret=q[l[0]][0]<q[l[1]][1]?q[l[0]++][0]+1:q[l[1]++][1]+1;
110        ans=max(ans,i-ret+1);
111     }
112     cout<<ans<<endl;
113
114     return 0;
115
116 }  

2500: 幸福的道路

Time Limit: 20 Sec  Memory Limit: 256 MB
Submit: 113  Solved: 49
[Submit][Status]

Description

小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光.

他们画出了晨练路线的草图,眼尖的小T发现可以用树来描绘这个草图.

他们不愿枯燥的每天从同一个地方开始他们的锻炼,所以他们准备给起点标号
后顺序地从每个起点开始(第一天从起点一开始,第二天从起点二开始……).
而且他们给每条道路定上一个幸福的值.很显然他们每次出发都想走幸福值和最长的路线(即从起点到树上的某一点路径中最长的一条).

他们不愿再经历之前的大起大落,所以决定连续几天的幸福值波动不能超过M(即一段连续的区间并且区间的最大值最小值之差不超过M).他们想知道要是这样的话他们最多能连续锻炼多少天(hint:不一定从第一天一直开始连续锻炼)?

现在,他们把这个艰巨的任务交给你了!

Input

第一行包含两个整数N, M(M<=10^9).

第二至第N行,每行两个数字Fi , Di, 第i行表示第i个节点的父亲是Fi,且道路的幸福值是Di.

Output

最长的连续锻炼天数

Sample Input

3 2
1 1
1 3

Sample Output

3
数据范围:
50%的数据N<=1000
80%的数据N<=100 000
100%的数据N<=1000 000

时间: 2025-01-01 10:19:48

BZOJ2500: 幸福的道路的相关文章

bzoj2500幸福的道路 树形dp+单调队列

2500: 幸福的道路 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 434  Solved: 170[Submit][Status][Discuss] Description 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光. 他们画出了晨练路线的草图,眼尖的小T发现可以用树来描绘这个草图. 他们不愿枯燥的每天从同一个地方开始他们的锻炼,所以他们准备给起点标号后顺序地从每个起点开始(第

bzoj2500: 幸福的道路(树形dp+单调队列)

好题.. 先找出每个节点的树上最长路 由树形DP完成 节点x,设其最长路的子节点为y 对于y的最长路,有向上和向下两种情况: down:y向子节点的最长路g[y][0] up:x的次长路的g[x][1]+dis[x][y] up:up[fa[x]]+dis[x][y] dfs1找向下,即向子节点的最长路 dfs2找向上的最长路 最后最长路f[i]=max(up[x],g[x][0]) 第二部分 找最长连续子序列,使得序列中abs(mx-mn)<=m 这次学习了用单调队列的做法 两个队列mx,mn

【BZOJ2500】幸福的道路 树形DP+RMQ+双指针法

[BZOJ2500]幸福的道路 Description 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光. 他们画出了晨练路线的草图,眼尖的小T发现可以用树来描绘这个草图. 他们不愿枯燥的每天从同一个地方开始他们的锻炼,所以他们准备给起点标号后顺序地从每个起点开始(第一天从起点一开始,第二天从起点二开始……). 而且他们给每条道路定上一个幸福的值.很显然他们每次出发都想走幸福值和最长的路线(即从起点到树上的某一点路径中最长的一条). 他

BZOJ 2500 幸福的道路

题面: 2500: 幸福的道路 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 366  Solved: 144[Submit][Status][Discuss] Description 小T与小L终于决定走在一起,他们不想浪费在一起的每一分每一秒,所以他们决定每天早上一同晨练来享受在一起的时光. 他们画出了晨练路线的草图,眼尖的小T发现可以用树来描绘这个草图. 他们不愿枯燥的每天从同一个地方开始他们的锻炼,所以他们准备给起点标号后顺序地从每个起点

【BZOJ】【2500】幸福的道路

树形DP+单调队列优化DP 好题(也是神题……玛雅我实在是太弱了TAT,真是一个250) 完全是抄的zyf的……orz我还是退OI保平安吧 第一步对于每一天求出一个从第 i 个点出发走出去的最长链的长度,树形DP解决…… g[x][0]表示从x的子树中,x到叶子的最长链,g[x][1]表示次长链.(用儿子更新父亲) f[x]表示从x向上走到某个父亲,再向下的最长链.(用父亲更新儿子) 这个DP是通过两次从根出发的dfs实现的. 那么我们现在就得到了a[i]=max(f[i],g[i][0])表示

BZOJ 2500 幸福的道路(race) 树上直径+平衡树

structHeal { priority_queue<int> real; priority_queue<int> stack; voidpush(int x) { real.push(x); } voidpop(int x) { stack.push(x); } inttop() { while(real.empty()==0&&stack.empty()==0&&real.top()==stack.top()) real.pop(),stack

BZOJ 2500 幸福的道路 树形DP+单调队列

题目大意:给定一棵树,令a[i]为从第i个节点出发的最长链,求a[i]中最长的区间,满足区间内最大值与最小值之差不超过m 读错题害死人,脑残害死人 求a[i]显然是树形DP 考虑从一个点出发的链可以从子节点走,也可以从父节点走 因此我们DP两次,第一次求出从子节点走的最长链,第二次求出从父节点走的最长链,两次取max就是答案 但是直接DP会有问题,因为从父节点走的最长链可能是从自己的子树出发的,这样就会走重 因此除记录从子节点出发的最长链外还要记录一个从另一个子节点出发的次长链,如果最长链长度相

cong

Directions:  Study the following cartoon carefully and write an essay in which you should 1) describe the cartoon, 2) interpret its meaning, and 3) point out its implications in our life. You should write about 160—200 words neatly on ANSWER SH

世界上最伟大的推销员

世界上最伟大的推销员 作者:奥格·曼狄诺 前言 <世界上最伟大的推销员>重在励志,通过对十个羊皮卷的反复诵读,培养人的 潜意识.长时间的坚持后,潜意识便转化为有益终生的好习惯.在日常生活和工作 中好的习惯是做任何事情获得成功的关键. 下面,简要摘录一些该书的经典话语 用来做为介绍: 今天我开始新的生活. 我郑重的发誓,绝不让任何事情妨碍我新生活的成长. 在阅读这些羊皮 卷的时候,我绝不浪费一天的时间,因为时光一去不返,失去的日子是无法弥补的.我也绝不打破 每天阅读的习惯.事实上,每天在这些新习