HDU 4008 Parent and son
树形DP+LCA+思路。这题可真是有点难度......所以准备详细写一下题解。
题意:给一颗无根树,有Q次询问,每次询问指定一个根节点X,然后让你计算Y节点的儿子和子孙中,编号最小的节点是多少。
我们先以1为根节点进行一次树形DP,记录下每个节点的儿子和子孙中,编号最小的节点是多少。
首先很容易想到一种情况:就是X和Y的最近公共祖先不是Y,这个时候,结果和以1为根节点建树是一模一样的,因为把X提到最上面去,不会影响到Y的子树的情况。
剩余的情况就是X和Y的最近公共祖先等于Y(这种情况就是X在Y的子树上),这个时候,如果把X提上去,那么无法直接从以1为根节点DP的结果得出答案,我们需要进行推导。这个时候,需要分两种情况。若Y==1,这个时候把X提上去,就好比从1连出来那么多的节点的子树中,X所在子树不再是Y的后代,如果X所在子树是最小值存在的子树,那么输出次小值,否则输出最小值。若Y!=1,这个时候,也是X所在子树不成为Y的后代,那么依然需要知道X所在子树是否是最小值存在的子树,还需要知道Y的父亲的情况,因为这个时候,Y的父亲那边的所有点成为了Y的子树。
综上所述,树形DP进行DFS的时候,我们需要记录每个子树的儿子最小值、次小值,后代最小值、次小值,每个节点的父亲节点的编号。
还需要o(1)判断两点是否在一条链上,这个可以通过时间戳来判断。
有了上述信息,在进行完先以1为根节点的DP之后,每一次询问,都可以在o(1)内得到答案。
代码写的又臭又长....还是不要看了。
#include <stdio.h>
#include <algorithm>
#include <string.h>
#include <queue>
#include <stack>
#include <map>
#include <vector>
using namespace std; const int maxn=+;
const int INF=0x7fffffff; int n,m;
vector<int>tree[maxn];
bool vis[maxn];
int MinSon[maxn],CiSon[maxn];
int MinDes[maxn],CiDes[maxn];
int belong[maxn];
int Fa;
int MinSonPos;
int MinDesPos;
int CiSonPos;
int CiDesPos;
struct P
{
int a,b;
}pp[maxn];
bool cmp(const P&a,const P&b) { return a.b<b.b;} int par[maxn];
int dfnIn[maxn],dfnOut[maxn];
int time;
int fa[maxn],l[maxn],p[maxn][];
int top, sun;
struct Edge{
int v;
Edge* next;
}*adj[maxn], edge[maxn << ]; void add(int u, int v){
Edge* p = &edge[++top];
p -> v = v;
p -> next = adj[u];
adj[u] = p;
}
void DFS(int u, int father, int depth){ fa[u] = father; l[u] = depth;
for(Edge* p = adj[u]; p; p = p -> next){
int v = p -> v;
if(v == father) continue; DFS(v, u, depth + ); }
}
void init_p(int n){
int log = ;
for(; ( << log) < n; log++); log--;
for(int i = ; i <= n; i++){
for(int j = ; j <= log; j++) p[i][j] = -;
}
for(int i = ; i <= n; i++) p[i][] = fa[i];
for(int j = ; j <= log; j++){
for(int i = ; i <= n; i++) if(p[i][j - ] != -) p[i][j] = p[p[i][j - ]][j - ];
}
}
int query(int a, int b){
int log, i;
if(l[a] < l[b]) swap(a, b);
for(log = ; ( << log) <= l[a]; log++); log--;
for(i = log; i >= ; i--){
if(l[a] - ( << i) > l[b]){
a = p[a][i];
}
}
sun = a;
if(fa[a] == b) return b;
a = fa[a];
for(i = log; i >= ; i--)
if(p[a][i] != - && p[a][i] != p[b][i]){
a = p[a][i]; b = p[b][i];
}
return fa[a];
} void init()
{
top = ;time=;
for(int i = ; i <= n; i++) adj[i] = ;
memset(vis,,sizeof vis);
for(int i=;i<=n;i++) MinSon[i]=INF,MinDes[i]=INF;
for(int i=;i<=n;i++) CiSon[i]=INF,CiDes[i]=INF;
for(int i=;i<=n;i++) tree[i].clear();
} void read()
{
for(int i=; i<n; i++)
{
int u,v;
scanf("%d%d",&u,&v);
add(u,v);
add(v,u);
tree[u].push_back(v);
tree[v].push_back(u);
}
} void dfs(int now)
{
dfnIn[now]=++time;
if(now!=) belong[now]=Fa; bool fail=;
for(int i=;i<tree[now].size();i++)
if(!vis[tree[now][i]]) fail=; if(fail)
{
dfnOut[now]=++time;
return;
}
int *ff= new int[tree[now].size()+];
for(int i=;i<tree[now].size();i++) ff[i]=;
for(int i=;i<tree[now].size();i++)
{
if(vis[tree[now][i]]) continue;
ff[i]=; par[tree[now][i]]=now;
if(now==) Fa=tree[now][i];
vis[tree[now][i]]=;
dfs(tree[now][i]);
MinSon[now]=min(MinSon[now],tree[now][i]);
MinDes[now]=min(MinDes[now],MinDes[tree[now][i]]);
}
MinDes[now]=min(MinDes[now],MinSon[now]); for(int i=;i<tree[now].size();i++)
{
if(!ff[i]) continue;
if(tree[now][i]!=MinSon[now])
CiSon[now]=min(CiSon[now],tree[now][i]); if(MinDes[tree[now][i]]!=MinDes[now])
CiDes[now]=min(CiDes[now],MinDes[tree[now][i]]); if(CiDes[tree[now][i]]!=MinDes[now])
CiDes[now]=min(CiDes[now],CiDes[tree[now][i]]); }
if(MinSon[now]!=MinDes[now])
CiDes[now]=min(CiDes[now],MinSon[now]); dfnOut[now]=++time; delete []ff;
} void work()
{
memset(vis,,sizeof vis);
DFS(, -, );
init_p(n);
vis[]=;dfs(); MinSonPos=INF; MinDesPos=INF;
CiSonPos=INF; CiDesPos=INF; for(int i=;i<tree[].size();i++)
MinSonPos=min(MinSonPos,tree[][i]); for(int i=;i<tree[].size();i++)
{
if(tree[][i]==MinSonPos) continue;
CiSonPos=min(CiSonPos,tree[][i]);
} int yy=;
for(int i=;i<tree[].size();i++)
{
pp[yy].a=tree[][i];
pp[yy++].b=min(tree[][i],MinDes[tree[][i]]);
}
sort(pp,pp+yy,cmp);
MinDesPos=pp[].a;
if(yy>=) CiDesPos=pp[].a; for(int i=;i<=m;i++)
{
int X,Y;
scanf("%d%d",&X,&Y);
int U=query(X,Y); if(U!=Y)
{
if(MinDes[Y]==INF) printf("no answers!\n");
else printf("%d %d\n",MinSon[Y],MinDes[Y]);
} else
{
if(Y==)
{
int ans1,ans2;
int be=belong[X];
if(MinSonPos==be) ans1=CiSonPos;
else ans1=MinSonPos;
if(MinDesPos==be) ans2=CiDesPos;
else ans2=MinDesPos;
if(ans1==INF||ans2==INF) printf("no answers!\n");
else printf("%d %d\n",ans1,min(ans2,MinDes[ans2]));
}
else
{
int ans=INF;
if(dfnIn[X]>=dfnIn[MinSon[Y]]&&dfnOut[X]<=dfnOut[MinSon[Y]]) ans=min(CiSon[Y],par[Y]);
else if(dfnIn[MinSon[Y]]>=dfnIn[X]&&dfnOut[MinSon[Y]]<=dfnOut[X]) ans=min(CiSon[Y],par[Y]);
else ans=min(MinSon[Y],par[Y]);
printf("%d 1\n",ans);
}
}
}
} int main()
{
int T; scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&m);
init();
read();
work();
printf("\n");
}
return ;
}
HDU 4008 Parent and son的更多相关文章
- HDU 4008 Parent and son LCA+树形dp
题意: 给定case数 给定n个点的树,m个询问 以下n-1行给出树边 m个询问 x y 问:以x为根.y子树下 y的最小点标的儿子节点 和子孙节点 思路: 用son[u][0] 表示u的最小儿子 s ...
- HDU4008 Parent and son(树形DP LCA)
先记录以1为根时每个节点子树儿子节点的最大与次小值,询问x, y时,先判断x在不在y的子树范围内,若不在,结果为y的儿子结点,后继的最小值. 若x在y的子树范围内,若y儿子最小值是x的前驱,从次小值与 ...
- hdu 4008 树形dp
思路:我们定义一个dfn[i],Maxndfn[i]来确定节点i的访问次序,以及其子节点的最大访问次序.那么另一个节点是其子树的节点当且仅当dfn[j]>=dfn[i]&&dfn ...
- Parent and son
Give you a tree with N vertices and N‐ 1 edges, and then ask you Q queries on “which vertex is Y's s ...
- [c++] Smart Pointers
内存管理方面的知识 基础实例: #include <iostream> #include <stack> #include <memory> using names ...
- Python之路【第十八篇】Django小项目简单BBS论坛部分内容知识点
开发一个简单的BBS论坛 项目需求: 整体参考“抽屉新热榜” + “虎嗅网” 实现不同论坛版块 帖子列表展示 帖子评论数.点赞数展示 在线用户展示 允许登录用户发贴.评论.点赞 允许上传文件 帖子可被 ...
- python 学习笔记二十 django项目bbs论坛
项目:开发一个简单的BBS论坛 需求: 整体参考“抽屉新热榜” + “虎嗅网” 实现不同论坛版块 帖子列表展示 帖子评论数.点赞数展示 在线用户展示 允许登录用户发贴.评论.点赞 允许上传文件 帖子可 ...
- Java(接口与继承)动手动脑
1>继承条件下的构造方法调用 运行 TestInherits.java 示例,观察输出,注意总结父类与子类之间构造方法的调用关系修改 Parent 构造方法的代码,显式调用 GrandParen ...
- JS魔法堂:判断节点位置关系
一.前言 在polyfill querySelectorAll 和写弹出窗时都需要判断两个节点间的位置关系,通过jQuery我们可以轻松搞定,但原生JS呢?下面我将整理各种判断方法,以供日后查阅. 二 ...
随机推荐
- IOS NSArray 倒序
NSMutableArray *array = [NSMutableArray arrayWithObjects:@"1",@"2",@"3" ...
- curl file_get_contents fsockopen
三种处理的方式: curl file_get_contents fsockopen fsockopen 是比较底层的调用,属于网络系统的socket调用,而curl经过的包装支 ...
- cmd下载文件
进入cmd 输入ftp 192.168.1.200 然后按照提示输入用户名和密码 cd 进入要下载的目录 dir 看操作权限 lcd查看本地要装下载文件的目录 prompt关闭交互模式 mget da ...
- 添加以及删除className
<!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type" content ...
- java 数据流
Example10_11.java import java.io.*; public class Example10_11 { public static void main(String args[ ...
- 学习笔记——原型模式Prototype
原型模式,简单说就是具有一个克隆方法,外部可以直接使用此方法得到相应对象的拷贝对象. 比如哆啦A梦的复制镜,一照,就把物品拷贝了一份(虽然是镜子复制是相反的,这里就忽略这个细节了) C++中依靠拷贝构 ...
- Ubuntu + Django + Nginx + uwsgi
环境 Ubuntu 14.04 Python 2.7 Django 1.8.4 1 安装Nginx sudo apt-get install nginx 测试 sudo /etc/init. ...
- MQ日常维护操作手册
假设队列管理器为QMgrName,以下所有使用QMgrName的地方您都可以替换成您维护的mq队列管理器名称. 一.MQ的启动与停止 用root用户启/停需要root用户包含在mqm组中. 1.MQ的 ...
- ARM汇编指令的一些总结-转
ARM汇编指令的一些总结ARM汇编指令很多,但是真正常用的不是很多,而且需要认真琢磨的又更少了.比较有用的是MOV B BL LDR STR还是通过具体汇编代码来学习吧.@ disable watch ...
- Hadoop 文本分类 终于跑通了
Training 入口 package org.wordCount; import java.util.ArrayList; import java.util.List; import org.apa ...