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=100000+10;
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][25];
 int top, sun;
 struct Edge{
     int v;
     Edge* next;
 }*adj[maxn], edge[maxn << 1];

 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 + 1);

      }
 }
 void init_p(int n){
      int log = 0;
      for(; (1 << log) < n; log++); log--;
      for(int i = 1; i <= n; i++){
          for(int j = 0; j <= log; j++)  p[i][j] = -1;
      }
      for(int i = 1; i <= n; i++) p[i][0] = fa[i];
      for(int j = 1; j <= log; j++){
          for(int i = 1; i <= n; i++) if(p[i][j - 1] != -1) p[i][j] = p[p[i][j - 1]][j - 1];
      }
 }
 int query(int a, int b){
     int log, i;
     if(l[a] < l[b]) swap(a, b);
     for(log = 1; (1 << log) <= l[a]; log++); log--;
     for(i = log; i >= 0; i--){
         if(l[a] - (1 << i) > l[b]){
             a = p[a][i];
         }
     }
     sun = a;
     if(fa[a] == b) return b;
     a = fa[a];
     for(i = log; i >= 0; i--)
         if(p[a][i] != -1 && p[a][i] != p[b][i]){
             a = p[a][i]; b = p[b][i];
         }
     return fa[a];
 }

void init()
{
    top = 0;time=0;
    for(int i = 1; i <= n; i++) adj[i] = 0;
    memset(vis,0,sizeof vis);
    for(int i=1;i<=n;i++) MinSon[i]=INF,MinDes[i]=INF;
    for(int i=1;i<=n;i++) CiSon[i]=INF,CiDes[i]=INF;
    for(int i=1;i<=n;i++) tree[i].clear();
}

void read()
{
    for(int i=1; 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!=1) belong[now]=Fa;

    bool fail=1;
    for(int i=0;i<tree[now].size();i++)
        if(!vis[tree[now][i]]) fail=0;

    if(fail)
    {
        dfnOut[now]=++time;
        return;
    }
    int *ff= new int[tree[now].size()+5];
    for(int i=0;i<tree[now].size();i++) ff[i]=0;
    for(int i=0;i<tree[now].size();i++)
    {
        if(vis[tree[now][i]]) continue;
        ff[i]=1; par[tree[now][i]]=now;
        if(now==1) Fa=tree[now][i];
        vis[tree[now][i]]=1;
        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=0;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,0,sizeof vis);
    DFS(1, -1, 0);
    init_p(n);
    vis[1]=1;dfs(1);
    
    MinSonPos=INF; MinDesPos=INF;
    CiSonPos=INF; CiDesPos=INF;

    for(int i=0;i<tree[1].size();i++)
        MinSonPos=min(MinSonPos,tree[1][i]);

    for(int i=0;i<tree[1].size();i++)
    {
        if(tree[1][i]==MinSonPos) continue;
        CiSonPos=min(CiSonPos,tree[1][i]);
    }

    int yy=0;
    for(int i=0;i<tree[1].size();i++)
    {
        pp[yy].a=tree[1][i];
        pp[yy++].b=min(tree[1][i],MinDes[tree[1][i]]);
    }
    sort(pp,pp+yy,cmp);
    MinDesPos=pp[0].a;
    if(yy>=1) CiDesPos=pp[1].a;


    for(int i=1;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==1)
            {
                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 0;
}

 

posted @ 2016-02-12 21:03  Fighting_Heart  阅读(326)  评论(0编辑  收藏  举报