hdu 4123 树形DP+RMQ
http://acm.hdu.edu.cn/showproblem.php?
he requires that every participant must start from a different house and run AS FAR AS POSSIBLE without passing a road more than once. The distance difference between the one who runs the longest distance and the one who runs the shortest distance is called
“race difference” by Bob. Bob does not want the “race difference”to be more than Q. The houses are numbered from 1 to N. Bob wants that the No. of all starting house must be consecutive. He is now asking you for help. He wants to know the maximum number of
starting houses he can choose, by other words, the maximum number of people who can take part in his race.
The first line of each test case contains two integers N and M. N is the number of houses, M is the number of queries.
The following N-1 lines, each contains three integers, x, y and z, indicating that there is a road of length z connecting house x and house y.
The following M lines are the queries. Each line contains an integer Q, asking that at most how many people can take part in Bob’s race according to the above mentioned rules and under the condition that the“race difference”is no more than Q.
The input ends with N = 0 and M = 0.
(N<=50000 M<=500 1<=x,y<=N 0<=z<=5000 Q<=10000000)
5 5 1 2 3 2 3 4 4 5 3 3 4 2 1 2 3 4 5 0 0
1 3 3 3 5
- /**
- hdu 4123 树形DP+RMQ
- 题目大意:给定一棵树,每个点都从当前位置走到距离最远的位置。1~n的连续区间中最大而且走的最远距离差值不超过Q的区间右多大
- 解题思路:两遍dfs求出在树中到当前点的最长距离:dfs1求出以当前节点为根节点的子树中到该节点的最长距离和次长距离,dfs2将上一步求出的
- 最长距离和经过根节点的最长距离比較取最大。利用RMQ查询寻找最大区间
- */
- #include <stdio.h>
- #include <string.h>
- #include <algorithm>
- #include <iostream>
- using namespace std;
- const int N=50050;
- int head[N],ip;
- struct note
- {
- int v,w,next;
- }edge[N*2];
- void init()
- {
- memset(head,-1,sizeof(head));
- ip=0;
- }
- void addedge(int u,int v,int w)
- {
- edge[ip].v=v,edge[ip].w=w,edge[ip].next=head[u],head[u]=ip++;
- }
- int maxn[N],smaxn[N],maxid[N],smaxid[N];
- void dfs1(int u,int pre)
- {
- maxn[u]=smaxn[u]=maxid[u]=smaxid[u]=0;
- for(int i=head[u];i!=-1;i=edge[i].next)
- {
- int v=edge[i].v;
- if(pre==v)continue;
- dfs1(v,u);
- if(maxn[v]+edge[i].w>smaxn[u])
- {
- smaxid[u]=v;
- smaxn[u]=maxn[v]+edge[i].w;
- if(maxn[u]<smaxn[u])
- {
- swap(maxn[u],smaxn[u]);
- swap(maxid[u],smaxid[u]);
- }
- }
- }
- }
- void dfs2(int u,int pre)
- {
- for(int i=head[u];i!=-1;i=edge[i].next)
- {
- int v=edge[i].v;
- if(pre==v)continue;
- if(maxid[u]==v)
- {
- if(smaxn[u]+edge[i].w>smaxn[v])
- {
- smaxn[v]=smaxn[u]+edge[i].w;
- smaxid[v]=u;
- if(maxn[v]<smaxn[v])
- {
- swap(maxn[v],smaxn[v]);
- swap(maxid[v],smaxid[v]);
- }
- }
- }
- else
- {
- if(maxn[u]+edge[i].w>smaxn[v])
- {
- smaxn[v]=maxn[u]+edge[i].w;
- smaxid[v]=u;
- if(maxn[v]<smaxn[v])
- {
- swap(maxn[v],smaxn[v]);
- swap(maxid[v],smaxid[v]);
- }
- }
- }
- dfs2(v,u);
- }
- }
- int a[N],n,m;
- int dp1[N][30];
- int dp2[N][30];
- void RMQ_init(int n)
- {
- for(int i=1;i<=n;i++)
- {
- dp1[i][0]=a[i];
- dp2[i][0]=a[i];
- }
- for(int j=1;(1<<j)<=n;j++)
- {
- for(int i=1;i+(1<<j)-1<=n;i++)///白书上的模板,次行稍作修改。否则dp数组要扩大一倍防止RE
- {
- dp1[i][j]=max(dp1[i][j-1],dp1[i+(1<<(j-1))][j-1]);
- dp2[i][j]=min(dp2[i][j-1],dp2[i+(1<<(j-1))][j-1]);
- }
- }
- }
- int rmq(int x,int y)
- {
- int k=0;
- while((1<<(k+1))<=y-x+1)k++;
- return max(dp1[x][k],dp1[y-(1<<k)+1][k])-min(dp2[x][k],dp2[y-(1<<k)+1][k]);
- }
- int main()
- {
- while(~scanf("%d%d",&n,&m))
- {
- if(n==0&&m==0)break;
- init();
- for(int i=1;i<n;i++)
- {
- int u,v,w;
- scanf("%d%d%d",&u,&v,&w);
- addedge(u,v,w);
- addedge(v,u,w);
- }
- dfs1(1,-1);
- dfs2(1,-1);
- for(int i=1;i<=n;i++)
- {
- a[i]=maxn[i];
- }
- RMQ_init(n);
- while(m--)
- {
- int Q;
- scanf("%d",&Q);
- int ans=0;
- int id=1;
- for(int i=1;i<=n;i++)
- {
- while(id<=i&&rmq(id,i)>Q)id++;
- ans=max(ans,i-id+1);
- }
- printf("%d\n",ans);
- }
- }
- return 0;
- }
hdu 4123 树形DP+RMQ的更多相关文章
- hdu 4123 树形DP+单调队列
http://acm.hust.edu.cn/vjudge/problem/25790 这题基本同poj 3162 要注意mx,mx2,vx,vx2每次都要初始化 #include <iostr ...
- 树形DP+RMQ+尺取法 hdu4123
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4123 参考博客:两种解法-树形dp+二分+单调队列(或RMQ)-hdu-4123-Bob’s Race ...
- HDU 1520 树形dp裸题
1.HDU 1520 Anniversary party 2.总结:第一道树形dp,有点纠结 题意:公司聚会,员工与直接上司不能同时来,求最大权值和 #include<iostream> ...
- HDU 1561 树形DP入门
The more, The Better Time Limit: 6000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Oth ...
- HDU 2196树形DP(2个方向)
HDU 2196 [题目链接]HDU 2196 [题目类型]树形DP(2个方向) &题意: 题意是求树中每个点到所有叶子节点的距离的最大值是多少. &题解: 2次dfs,先把子树的最大 ...
- HDU 1520 树形DP入门
HDU 1520 [题目链接]HDU 1520 [题目类型]树形DP &题意: 某公司要举办一次晚会,但是为了使得晚会的气氛更加活跃,每个参加晚会的人都不希望在晚会中见到他的直接上司,现在已知 ...
- codevs 1380/HDU 1520 树形dp
1380 没有上司的舞会 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 查看运行结果 回到问题 题目描述 Description Ural大学有N个职员 ...
- HDU 5834 [树形dp]
/* 题意:n个点组成的树,点和边都有权值,当第一次访问某个点的时候获得利益为点的权值 每次经过一条边,丢失利益为边的权值.问从第i个点出发,获得的利益最大是多少. 输入: 测试样例组数T n n个数 ...
- hdu 4267 树形DP
思路:先dfs一下,找出1,n间的路径长度和价值,回溯时将该路径长度和价值清零.那么对剩下的图就可以直接树形dp求解了. #include<iostream> #include<al ...
随机推荐
- SGU 133.Border
水题不说了 #include <iostream> #include <cstring> #include <cstdio> #include <cmath& ...
- 那些年,我们一起被坑的H5音频
原文地址:http://weibo.com/p/23041874d6cedd0102vkbr 不要被这么文艺的标题吓到,这里不会跟你讲述中学时期泡妞史,也不会有其它什么现实不该有而小说噼里啪啦不能 ...
- linux之uniq
Linux命令uniq的作用是过滤重复部分显示文件内容,这个命令读取输入文件,并比较相邻的行.在正常情况下,第二个及以后更多个重复行将被删去,行 比较是根据所用字符集的排序序列进行的.该命令加工后的结 ...
- ng-class css样式
<style> .error{background-color: red;} .warning{background-color: yellow;} </style> < ...
- android:persistent属性
application PhoneApp既没有被Broadcast唤醒,也没有被其他service调用,那么是android是通过什么方式来启动PhoneApp,所以就发现了属性android:per ...
- 黑马程序员——利用swap函数研究C的指针
------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 设计3个函数,分别实现已下功能: 交换两个整数 交换两个整形指针 交换任意两个同类型的变量 #i ...
- NandFlash
一.概述 1.NandFlash NAND结构能提供极高的单元密度,可以达到高存储密度,比如能达到256M,并且写入和擦除的速度也很快.应用NAND的困难在于flash的管理需要特殊的系统接口. 2. ...
- NSSet与NSArray区别
NSSet与NSArray区别 NSSet到底什么类型,其实它和NSArray功能性质一样,用于存储对象,属于集合: NSSet , NSMutableSet类声明编程接口对象,无序的集合, ...
- 应用ubuntu(安装)
U盘安装Ubuntu 12.04. 工具 UltraISO 9.6.1 ubuntu-12.04.3-desktop-i386 启动U盘 安装UltralISO,启动 文件—打开,选中下载的ubunt ...
- XML CDATA
/* <![CDATA[ */var mv_dynamic_to_top = {"text":"To Top","version":& ...