zoukankan      html  css  js  c++  java
  • 浅谈关于树形dp求树的直径问题

    在一个有n个节点,n-1条无向边的无向图中,求图中最远两个节点的距离,那么将这个图看做一棵无根树,要求的即是树的直径。####

    求树的直径主要有两种方法:树形dp和两次bfs/dfs,因为我太菜了不会写后者这里只介绍树形dp

    • 树形dp求树的直径
      我们不妨设1号点为根节点,那么这就可以看做一棵有根树。
      设D[x]表示从节点x出发,往以x为根的子树走,能够到达的最远距离。设x的子节点分别为(y_1,y_2,y_3,...,y_t)(edge(x,y))表示从x到y的边权,则可以得到状态转移方程:
      (D[x]={(D[y_i]+edge(x,y_i))}_{max})
      接下来,我们考虑对于每个节点x求出经过x的最长链的长度F[x],整棵树的直径就是max{F[x]}(1<=x<=n)。

      现在我们考虑如何求F[x]。
      对于任意两个节点yi和yj,经过节点x的最长链的长度可以通过四个部分来构成:

      • D[yi]
      • D[yj]
      • 从x到yi的距离
      • 从x到yj的距离

      不妨设j<i,则有:

    (F[x]= {(D[y_i]+D[y_j]+edge(x,y_i)+edge(x,y_j))}_{max})

    对应代码如下:

    void dp(int x){
    	v[x]=1;	
    	for(register int i=head[x];i;i=nxt[i]){
    		int y=ver[i];
    		if(v[y])continue;	
    		dp(y); 
    		ans=max(ans,d[x]+d[y]+edge[i]);
    		d[x]=max(d[x],d[y]+edge[i]);
    	}
    }
    

    代码解释可以看图:

    参考资料:李煜东《算法竞赛进阶指南》

  • 相关阅读:
    shell学习(15)- eval及shell No such file or directory解决办法
    30张图带你彻底理解红黑树
    linux中$@,$*,$0,$$,$?参数的含义
    QPS,TPS,吞吐量,响应时间详解及关系
    shell学习(14)- who
    SpringBoot 入门
    创建 Java 项目
    部分 GIT 命令
    Spark Executor Task 的执行和数量
    Ubuntu 18 单机安装 HDP 3
  • 原文地址:https://www.cnblogs.com/kma093/p/9742317.html
Copyright © 2011-2022 走看看