思路:我们定义一个dfn[i],Maxndfn[i]来确定节点i的访问次序,以及其子节点的最大访问次序。那么另一个节点是其子树的节点当且仅当dfn[j]>=dfn[i]&&dfn[j]<=Maxdfn[i];

这是就可以先以1为根,dfs找出所有节点最小儿子,次小儿子,以及最小子孙,次小子孙。

剩下就可以分情况:

对于x,y:

1.若与y相连的边数为1,那么一定就是no answers!。

2.对于最小儿子节点的询问:

(1)若x是y的子节点,那么如果x包含在y的最小儿子子树中,那么就答案就是次小儿子与其父节点中较小的,否则就是最小儿子与其父节点中较小的。

(2)若y是x点子节点(设不是y的子节点的都是其根节点),那么就直接输出最小儿子。

3.对于最小子孙节点的询问:

(1)若x是y的子节点,那么如果x包含在y的最小子孙所在子树中(小于以y为根的子树),那么就答案就是次小子孙,否则就是最小子孙,其实这只设和y为1的情况,若y为其它节点,其最小子孙直接就是1。

(2)若y是x点子节点(设不是y的子节点的都是其根节点),那么就直接输出最小子孙。

#include<iostream>
#include<cstring>
#include<algorithm>
#include<cstdio>
#include<queue>
#define Maxn 100010
#define inf 0x7fffffff
using namespace std;
int son[Maxn],nson[Maxn],der[Maxn],nder[Maxn],n,q,dfn[Maxn],Maxdfn[Maxn],lab,dp[Maxn],road[Maxn],father[Maxn];
vector<int> head[Maxn];
void init()
{
int i;
memset(son,,sizeof(son));
memset(nson,,sizeof(nson));
memset(der,,sizeof(der));
memset(nder,,sizeof(nder));
memset(dfn,,sizeof(dfn));
memset(father,,sizeof(father));
memset(dp,,sizeof(dp));
memset(road,,sizeof(road));
memset(Maxdfn,,sizeof(Maxdfn));
lab=;
for(i=;i<=n;i++)
head[i].clear();
}
void add(int u,int v)
{
head[u].push_back(v);
head[v].push_back(u);
}
void dfs(int u,int f)
{
int i,v,si;
dfn[u]=Maxdfn[u]=++lab;
dp[u]=u;
son[u]=nson[u]=der[u]=nder[u]=inf;
si=head[u].size();
for(i=;i<si;i++)
{
v=head[u][i];
if(v==f)
continue;
dfs(v,u);
dp[u]=min(dp[u],dp[v]);
Maxdfn[u]=max(Maxdfn[u],Maxdfn[v]);
if(v<son[u])
{
nson[u]=son[u];
son[u]=v;
}
else
{
if(v<nson[u])
nson[u]=v;
}
if(der[u]>dp[v])
{
nder[u]=der[u];
der[u]=dp[v];
road[u]=v;
}
else
if(nder[u]>dp[v]) nder[u]=dp[v];
}
father[u]=f;
}
int main()
{
int i,j,a,b,t,x,y;
scanf("%d",&t);
while(t--)
{
init();
scanf("%d%d",&n,&q);
for(i=;i<n;i++)
{
scanf("%d%d",&a,&b);
add(a,b);
}
dfs(,);
for(i=;i<=q;i++)
{
scanf("%d%d",&x,&y);
int ans1,ans2;
if(head[y].size()==)
{
puts("no answers!");
continue;
}
if(dfn[x]<dfn[y]||dfn[x]>Maxdfn[y])
printf("%d %d\n",son[y],der[y]);
else
{
if(dfn[x]>=dfn[son[y]]&&dfn[x]<=Maxdfn[son[y]])
ans1=nson[y];
else
ans1=son[y];
if(dfn[x]>=dfn[road[y]]&&dfn[x]<=Maxdfn[road[y]])
ans2=nder[y];
else
ans2=der[y];
if(y!=) ans1=min(ans1,father[y]),ans2=;
printf("%d %d\n",ans1,ans2);
}
}
printf("\n");
}
return ;
}

hdu 4008 树形dp的更多相关文章

  1. hdu 4123 树形DP+RMQ

    http://acm.hdu.edu.cn/showproblem.php? pid=4123 Problem Description Bob wants to hold a race to enco ...

  2. HDU 1520 树形dp裸题

    1.HDU 1520  Anniversary party 2.总结:第一道树形dp,有点纠结 题意:公司聚会,员工与直接上司不能同时来,求最大权值和 #include<iostream> ...

  3. HDU 1561 树形DP入门

    The more, The Better Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Oth ...

  4. HDU 2196树形DP(2个方向)

    HDU 2196 [题目链接]HDU 2196 [题目类型]树形DP(2个方向) &题意: 题意是求树中每个点到所有叶子节点的距离的最大值是多少. &题解: 2次dfs,先把子树的最大 ...

  5. HDU 1520 树形DP入门

    HDU 1520 [题目链接]HDU 1520 [题目类型]树形DP &题意: 某公司要举办一次晚会,但是为了使得晚会的气氛更加活跃,每个参加晚会的人都不希望在晚会中见到他的直接上司,现在已知 ...

  6. codevs 1380/HDU 1520 树形dp

    1380 没有上司的舞会 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 查看运行结果 回到问题 题目描述 Description Ural大学有N个职员 ...

  7. HDU 5834 [树形dp]

    /* 题意:n个点组成的树,点和边都有权值,当第一次访问某个点的时候获得利益为点的权值 每次经过一条边,丢失利益为边的权值.问从第i个点出发,获得的利益最大是多少. 输入: 测试样例组数T n n个数 ...

  8. hdu 4267 树形DP

    思路:先dfs一下,找出1,n间的路径长度和价值,回溯时将该路径长度和价值清零.那么对剩下的图就可以直接树形dp求解了. #include<iostream> #include<al ...

  9. hdu 4607 (树形DP)

    当时比赛的时候我们找出来只要求出树的最长的边的节点数ans,如果要访问点的个数n小于ans距离直接就是n-1 如果大于的话就是(n-ans)*2+ans-1,当时求树的直径难倒我们了,都不会树形dp ...

随机推荐

  1. 03 javadoc

    javadoc从程序源代码中抽取类.方法.成员等注释形成一个和源代码配套的API帮助文档 1.标签.命令格式: 2.使用方式: 2.1 dos命令行格式:javadoc XXX.java 2.2 ec ...

  2. datasnap的前世今生

    随着XE6,XE7,以及半年以后即将发布的XE8,DATASNAP将顺应跨平台的需要, 有可能的情况是这样的:XE8,DATASNAP写的中间件将可以运行在LINUX服务器上面. 大家都知道COM是W ...

  3. iOS异常捕获

    文章目录 一. 系统Crash 二. 处理signal 下面是一些信号说明 关键点注意 三. 实战 四. Crash Callstack分析 – 进⼀一步分析 五. demo地址 六. 参考文献 前言 ...

  4. 关于java的continue、break关键字用法

    一 明确两个概念 循环:是指按照规定次数重复执行某一操作的全过程:其关键语句有for. foreach.while.do while 迭代:是指循环过程中单次操作,1次循环由n次迭代构成 二 用法归纳 ...

  5. Unity3D之Mecanim动画系统学习笔记(四):Animation State

    动画的设置 我们先看看Animation Clip的一些设置: Loop time:动画是否循环播放. 下面出现了3个大致一样的选项: Root Transform Rotation:表示为播放动画的 ...

  6. [PoC]某B2B网站的一个反射型XSS漏洞

    Author: Charlie 个人微博:http://YinYongYou.com 转载请注明出处. 工作过程纯粹手贱,测试了一下.然后发现了这么一个东西.有心利用能造成大范围影响.如可以自由修改用 ...

  7. Cygwin解决Windows远程登录linux服务器

    下载地址http://www.cygwin.com/install.html 选择mirror.htnshost.com网站下载的比较快. 安装Cygwin(/X)需要选择的包: openssh(必选 ...

  8. ThinkPad指纹验证在win7无法使用的解决方法

    原先本本装window7 64bit 专业版(正版),但用着用着觉得 很不爽 ,反应特慢.所以决定对本本来次大换血,换成windows server 2008 R2.最后在装指纹验证的时候,使用超级管 ...

  9. hdu 5278 Geometric Progression 高精度

    Geometric Progression Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://bestcoder.hdu.edu.cn/contes ...

  10. TinyXML:一个优秀的C++ XML解析器[转]

    TinyXML:一个优秀的C++ XML解析器 读取和设置xml配置文件是最常用的操作,试用了几个C++的XML解析器,个人感觉TinyXML是使用起来最舒服的,因为它的API接口和Java的十分类似 ...