码迷,mamicode.com
首页 > 其他好文 > 详细

POJ 3162 Walking Race 树形DP+线段树

时间:2015-08-20 18:45:19      阅读:167      评论:0      收藏:0      [点我收藏+]

标签:

给出一棵树,编号为1~n,给出数m

漂亮mm连续n天锻炼身体,每天会以节点i为起点,走到离i最远距离的节点

走了n天之后,mm想到知道自己这n天的锻炼效果

于是mm把这n天每一天走的距离记录在一起,成为一段长度为n的数列

现在mm想要从这数列中选出一个连续的区间,要求这个区间的max-min<=m

输出最长的区间

 

做了一个下午

思路:

分成2个部分:

1.求出数列,即对于一棵树,求出每一个节点能到达的最远距离

2.对于这段数列,选出一个区间,使得区间的max-min<=m,并且使得区间长度尽量长。

 

对于第1部分,其实就是我昨天做的题目,HDU2196,使用树形DP

dp[i][1]:表示从i出发到i的子树,能到达的最远距离

dp[i][2]:表示从i出发到i的子树,能到达的第二远的距离(求dp[i][0]的时候需要用到)

dp[i][0]:表示从i出发,经过i的父亲节点,能到达的最远距离

4次dfs,分别求出:

0.节点的depth,以便把边权(u,v)转化为u,v中depth较大的节点的点权,cost[root]=0;

1.求出dp[i][1]和son[i],son[i]表示dp[i][1]是通过子节点son[i]这道路径得到的

2.求出dp[i][2]

3.求出dp[i][0]

完成第一部分

 

第二部分,前段时间也做过类似的题,HDU5289

5289要求的是数列中有多少个区间满足条件,我是用RMQ+二分

这道求的是满足条件的最长区间长度

我主要时间是花在了解决第二部分

 

我刚开始也是用RMQ+二分

发现mle了

因为5289的长度n<=1e5,而这道,n<=1e6,所以RMQ开了会mle

然后我就想改为用线段树代替RMQ的作用,用线段树维护区间的最大值和最小值也很方便,还可以省下很多空间

然后,tle了

原因:

这道题和5289不同的是,这道题不用枚举每一个区间,判断到底满不满足要求

只要求出一个满足的最长的就好啦

然后把枚举左端点+二分右端点的代码注释了

换了种方式:

用2个指针l,r,

若区间[l,r]满足条件,r++,同时更新答案

若不满足条件,l++,直到[l,r]满足条件

复杂度O(n)

这就是传说中的尺取法?

 

这过程我还犯了一个及其严重的错误:

求最大值的时候,没有把max_ret初始化为-inf

求最小值的时候,没有把min_ret初始化为inf

又花了不少时间

 

然后终于过了

 

技术分享
  1 #include<cstdio>
  2 #include<cstring>
  3 
  4 using namespace std;
  5 #define lson l,m,rt<<1
  6 #define rson m+1,r,rt<<1|1
  7 
  8 inline int max(int a,int b)
  9 {
 10     return a>b?a:b;
 11 }
 12 
 13 inline int min(int a,int b)
 14 {
 15     return a<b?a:b;
 16 }
 17 
 18 const int maxn=1e6+5;
 19 const int inf=0x3f3f3f3f;
 20 
 21 struct Edge
 22 {
 23     int to,next;
 24 };
 25 Edge edge[maxn<<1];
 26 int head[maxn];
 27 int tot;
 28 int son[maxn];
 29 int dp[maxn][3];
 30 int e[maxn][2];
 31 int dep[maxn];
 32 int cost[maxn];
 33 int seg_max[maxn<<2];
 34 int seg_min[maxn<<2];
 35 int max_ret;
 36 int min_ret;
 37 
 38 void init()
 39 {
 40     memset(head,-1,sizeof head);
 41     tot=1;
 42     memset(dep,0,sizeof dep);
 43     memset(dp,0,sizeof dp);
 44     memset(son,-1,sizeof son);
 45 }
 46 
 47 void addedge(int u,int v)
 48 {
 49     edge[tot].to=v;
 50     edge[tot].next=head[u];
 51     head[u]=tot++;
 52 }
 53 
 54 void solve(int n,int m);
 55 void dfs0(int u,int pre);
 56 void dfs1(int u,int pre);
 57 void dfs2(int u,int pre);
 58 void dfs3(int u,int pre);
 59 void build(int l,int r,int rt);
 60 void pushup(int rt);
 61 int query(int L,int R,int n);
 62 void query_ret(int L,int R,int l,int r,int rt);
 63 
 64 int main()
 65 {
 66     int n,m;
 67     while(~scanf("%d %d",&n,&m))
 68     {
 69         init();
 70         for(int i=2;i<=n;i++)
 71         {
 72             int u,v;
 73             scanf("%d %d",&e[i][0],&e[i][1]);
 74             addedge(i,e[i][0]);
 75             addedge(e[i][0],i);
 76         }
 77         solve(n,m);
 78     }
 79     return 0;
 80 }
 81 
 82 void solve(int n,int m)
 83 {
 84     dfs0(1,-1);
 85     cost[1]=0;
 86     for(int i=2;i<=n;i++)
 87     {
 88         if(dep[i]>dep[e[i][0]])
 89             cost[i]=e[i][1];
 90         else
 91             cost[e[i][0]]=e[i][1];
 92     }
 93     dfs1(1,-1);
 94     dfs2(1,-1);
 95     dfs3(1,-1);
 96     build(1,n,1);
 97     /*
 98     for(int i=1;i<=n;i++)
 99     {
100         printf("%d\n",max(dp[i][0],dp[i][1]));
101     }
102     */
103     /*
104     int ans=0;
105     for(int i=1;i<=n;i++)
106     {
107         int l=i,r=n;
108         while(r-l>1)
109         {
110             int mid=(l+r)>>1;
111             if(query(i,mid,n)>m)
112                 r=mid;
113             else
114                 l=mid;
115         }
116         if(query(i,r,n)<=m)
117             ans=max(ans,r-i+1);
118         else
119             ans=max(ans,l-i+1);
120     }
121     */
122 
123     int ans=0;
124     int l=1,r=1;
125     while(r<=n)
126     {
127         while(l<=r&&query(l,r,n)>m)
128         {
129             l++;
130         }
131         while(r<=n&&query(l,r,n)<=m)
132         {
133             ans=max(ans,r-l+1);
134             r++;
135         }
136     }
137 
138     printf("%d\n",ans);
139     return ;
140 }
141 
142 void dfs0(int u,int pre)
143 {
144     for(int i=head[u];~i;i=edge[i].next)
145     {
146         int v=edge[i].to;
147         if(v==pre)
148             continue;
149         dep[v]=dep[u]+1;
150         dfs0(v,u);
151     }
152 }
153 
154 void dfs1(int u,int pre)
155 {
156     for(int i=head[u];~i;i=edge[i].next)
157     {
158         int v=edge[i].to;
159         if(v==pre)
160             continue;
161         dfs1(v,u);
162         if(dp[v][1]+cost[v]>dp[u][1])
163         {
164             dp[u][1]=dp[v][1]+cost[v];
165             son[u]=v;
166         }
167     }
168 }
169 
170 void dfs2(int u,int pre)
171 {
172     for(int i=head[u];~i;i=edge[i].next)
173     {
174         int v=edge[i].to;
175         if(v==pre)
176             continue;
177         dfs2(v,u);
178         if(v==son[u])
179             continue;
180         if(dp[v][1]+cost[v]>dp[u][2])
181         {
182             dp[u][2]=dp[v][1]+cost[v];
183         }
184     }
185 }
186 
187 void dfs3(int u,int pre)
188 {
189     for(int i=head[u];~i;i=edge[i].next)
190     {
191         int v=edge[i].to;
192         if(v==pre)
193             continue;
194         if(v==son[u])
195         {
196             dp[v][0]=max(dp[u][0],dp[u][2])+cost[v];
197         }
198         else
199         {
200             dp[v][0]=max(dp[u][0],dp[u][1])+cost[v];
201         }
202         dfs3(v,u);
203     }
204 }
205 
206 void pushup(int rt)
207 {
208     seg_max[rt]=max(seg_max[rt<<1],seg_max[rt<<1|1]);
209     seg_min[rt]=min(seg_min[rt<<1],seg_min[rt<<1|1]);
210 }
211 
212 void build(int l,int r,int rt)
213 {
214     if(l==r)
215     {
216         seg_max[rt]=seg_min[rt]=max(dp[l][0],dp[l][1]);
217         return ;
218     }
219     int m=(l+r)>>1;
220     build(lson);
221     build(rson);
222     pushup(rt);
223 }
224 
225 int query(int L,int R,int n)
226 {
227     max_ret=-inf;
228     min_ret=inf;
229     query_ret(L,R,1,n,1);
230     return max_ret-min_ret;
231 }
232 
233 void query_ret(int L,int R,int l,int r,int rt)
234 {
235     if(L<=l&&R>=r)
236     {
237         max_ret=max(max_ret,seg_max[rt]);
238         min_ret=min(min_ret,seg_min[rt]);
239         return ;
240     }
241     int m=(l+r)>>1;
242     if(L<=m)
243         query_ret(L,R,lson);
244     if(R>m)
245         query_ret(L,R,rson);
246 }
View Code

 

POJ 3162 Walking Race 树形DP+线段树

标签:

原文地址:http://www.cnblogs.com/-maybe/p/4745930.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!