树的直径概念及求解

article/2025/10/7 0:04:55

文章目录

      • 1. 使用两次DFS求得树的直径
      • 2. 使用树形DP求得树的直径
      • 3. 性质
      • 4. 参考文献和习题



树上任意两节点之间最长的简单路径即为树的「直径」。显然,一棵树可以有多条直径,他们的长度相等。可以用两次 D F S / B F S DFS/BFS DFS/BFS 或者树形 D P DP DP 的方法在 O ( n ) O(n) O(n) 时间求出树的直径。



1. 使用两次DFS求得树的直径

两次 D F S / B F S DFS/BFS DFS/BFS 。第一次任意选一个点进行 D F S / B F S DFS/BFS DFS/BFS 找到离它最远的点,此点就是最长路的一个端点,再以此点进行 D F S / B F S DFS/BFS DFS/BFS ,找到离它最远的点,此点就是最长路的另一个端点,于是就找到了树的直径。

证明:
假设此树的最长路径是从 s s s t t t,我们选择的点为 u u u。反证法:假设搜到的点是 v v v

    1. v v v在这条最长路径上,那么 d i s [ u , v ] > d i s [ u , v ] + d i s [ v , s ] dis[u,v]>dis[u,v]+dis[v,s] dis[u,v]>dis[u,v]+dis[v,s],显然矛盾。
    1. v v v不在这条最长路径上,我们在最长路径上选择一个点为 p o po po,则 d i s [ u , v ] > d i s [ u , p o ] + d i s [ p o , t ] dis[u,v]>dis[u,po]+dis[po,t] dis[u,v]>dis[u,po]+dis[po,t],那么有 d i s [ s , v ] = d i s [ s , p o ] + d i s [ p o , u ] + d i s [ u , v ] > d i s [ s , p o ] + d i s [ p o , t ] = d i s [ s , t ] dis[s,v]=dis[s,po]+dis[po,u]+dis[u,v]>dis[s,po]+dis[po,t]=dis[s,t] dis[s,v]=dis[s,po]+dis[po,u]+dis[u,v]>dis[s,po]+dis[po,t]=dis[s,t],即 d i s [ s , v ] > d i s [ s , t ] dis[s,v]>dis[s,t] dis[s,v]>dis[s,t],矛盾。

也许你想说 u u u本身就在最长路径,或则其它的一些情况,但其实都能用类似于上面的反证法来证明的。
综上所述,你两次 D F S / B F S DFS/BFS DFS/BFS 就可以求出最长路径的两个端点和路径长度。

上述证明过程建立在所有路径均不为负的前提下。如果树上存在负权边,则上述证明不成立。故若存在负权边,则无法使用两次 D F S / B F S DFS/BFS DFS/BFS 的方式求解直径。


代码实现如下:

const int N = 10000 + 10;int n, c, d[N]; 
//n是树节点数,c是第一次dfs找到的直径端点
//d[n]是所选根节点到某个下标点的最大距离
vector<int> E[N];
//E[n]是节点之间的连通情况,即edges//寻找以u为起始点/根节点,fa为前驱节点的,到u的最大距离的点
//即求直径端点c的过程
void dfs(int u, int fa) {for (int v : E[u]) {if (v == fa) continue; //不走回头路d[v] = d[u] + 1; //边权值均为1if (d[v] > d[c]) c = v; //如果更新后距离u的距离更大,更新cdfs(v, u);}
}int main() {//初始化图scanf("%d", &n);for (int i = 1; i < n; i++) {int u, v;scanf("%d %d", &u, &v);E[u].push_back(v), E[v].push_back(u);}dfs(1, 0); //第一次dfs找寻直径端点cd[c] = 0, dfs(c, 0); //第二次dfs找寻另一端点,前驱节点还是定为0printf("%d\n", d[c]); //输出旧c到新c的距离,即为树的直径return 0;
}

如果需要求出一条直径上所有的节点,则可以在第二次 D F S DFS DFS 的过程中,记录每个点的前序节点,即可从直径的一端一路向前,遍历直径上所有的节点。



2. 使用树形DP求得树的直径

我们记录当 1 1 1 为树的根时,每个节点作为子树的根向下,所能延伸的最长路径长度 d 1 d_1 d1 与次长路径(与最长路径无公共边)长度 d 2 d_2 d2,那么直径就是对于每一个点,该点 d 1 + d 2 d_1 + d_2 d1+d2 能取到的值中的最大值。

树形 D P DP DP可以在存在负权边的情况下求解出树的直径。

const int N = 10000 + 10;int n, d = 0;
int d1[N], d2[N];
vector<int> E[N];void dfs(int u, int fa) {d1[u] = d2[u] = 0;for (int v : E[u]) {if (v == fa) continue;dfs(v, u);int t = d1[v] + 1;if (t > d1[u])d2[u] = d1[u], d1[u] = t;else if (t > d2[u])d2[u] = t;}d = max(d, d1[u] + d2[u]);
}int main() {scanf("%d", &n);for (int i = 1; i < n; i++) {int u, v;scanf("%d %d", &u, &v);E[u].push_back(v), E[v].push_back(u);}dfs(1, 0);printf("%d\n", d);return 0;}

如果需要求出一条直径上所有的节点,则可以在 D P DP DP 的过程中,记录下每个节点能向下延伸的最长路径与次长路径(定义同上)所对应的子节点,在求 d d d 的同时记下对应的节点 u u u,使得 d = d 1 [ u ] + d 2 [ u ] d = d_1[u] + d_2[u] d=d1[u]+d2[u],即可分别沿着从 u u u 开始的最长路径的次长路径对应的子节点一路向某个方向(对于无根树,虽然这里指定了 1 1 1 为树的根,但仍需记录每点跳转的方向;对于有根树,一路向上跳即可),遍历直径上所有的节点。



3. 性质

若树上所有边边权均为正,则树的所有直径中点重合

证明:使用反证法。设两条中点不重合的直径分别为 δ ( s , t ) \delta(s,t) δ(s,t) δ ( s ′ , t ′ ) \delta(s',t') δ(s,t),中点分别为 x x x x ′ x' x。显然, δ ( s , x ) = δ ( x , t ) = δ ( s ′ , x ′ ) = δ ( x ′ , t ′ ) \delta(s,x) = \delta(x,t) = \delta(s',x') = \delta(x',t') δ(s,x)=δ(x,t)=δ(s,x)=δ(x,t)

δ ( s , t ′ ) = δ ( s , x ) + δ ( x , x ′ ) + δ ( x ′ , t ′ ) > δ ( s , x ) + δ ( x , t ) = δ ( s , t ) \delta(s,t') = \delta(s,x) + \delta(x,x') + \delta(x',t') > \delta(s,x) + \delta(x,t) = \delta(s,t) δ(s,t)=δ(s,x)+δ(x,x)+δ(x,t)>δ(s,x)+δ(x,t)=δ(s,t),与 δ ( s , t ) \delta(s,t) δ(s,t) 为树上任意两节点之间最长的简单路径矛盾,故性质得证。



4. 参考文献和习题

OI_Wiki_树的直径
PT07Z - Longest path in a tree


http://chatgpt.dhexx.cn/article/Fn7jUHyx.shtml

相关文章

树的直径两种求法

首先先介绍一下什么是树的直径&#xff0c;树的直径就是树中所有最短路经距离的最大值。 求树的直径通常有两种方法&#xff0c;一种是通过两次搜索&#xff08;bfs和dfs均可&#xff09;&#xff0c;另一种就是通过树形dp来求了。 先来介绍一下用两次深搜来求树的直径&#x…

树的直径的概念

树的直径的定义: 在一棵树中&#xff0c;每一条边都有权值&#xff0c;树中的两个点之间的距离&#xff0c;定义为连接两点的路径上边权之和&#xff0c; 那么树上最远的两个点&#xff0c;他们之间的距离&#xff0c;就被称之为&#xff0c;树的直径。 树的直径的别称&#x…

树的直径

【定义】 我们将一棵树T ( V&#xff0c;E )的直径定义为maxδ ( u&#xff0c;v ) ( u&#xff0c;v ∈ V )&#xff0c;也就是说&#xff0c;树中所有最短路径距离的最大值即为树的直径。 【做法】 例题传送门Cow Marathon&#xff08;POJ 1985&#xff09; 对于树的直径…

浅谈树的直径

一、定义&#xff1a; 我们将一棵树T(V,E)的直径定义为max(u,v) (u,v∈V)&#xff0c;也就是说&#xff0c;树中所有最短路径距离的最大值即为树的直径。&#xff08;就是树中的最长路径的长度&#xff09; 二、求解树德直径&#xff1a; 求得树的直径有两种方法&#xff0c;一…

3.网络基础-三层路由网络

3.1、IP地址 初识IP地址 • 在IP网络中&#xff0c;通信节点需要有一个唯一的IP地址&#xff1b; • IP地址用于IP报文的寻址以及标识一个节点&#xff1b; • IPv4地址一共32bits&#xff0c;使用点分十进制的形式表示&#xff1b; IP地址的分类 E类是保留地址 公有IP及私有…

计算机网络——配置动态路由实验

配置动态路由实验 实验目的实验软件实验要求实验知识实验步骤实验结果 实验目的 掌握 RIP 协议配置。RIP 协议配置的命令为&#xff1a;router(config)#network <connected-network> 其中参数 <connected-network> 表示路由器的直连网络号。 实验软件 Cisco Pac…

路由技术基础

路由技术是在网络拓扑结构中为不同节点的数据提供传输路径的技术&#xff0c;路由选择算法是其核心内容。路由选择算法分为静态路由选择算法和动态路由选择算法。 一.路由基础 1.路由的基本概念 路由、路由器 &#xff08;1&#xff09;路由 路由是指导IP报文从源发送到目…

网络路由交换 -- 静态路由 和 缺省路由

1.IP路由基础 1.什么是静态路由&#xff1f; 静态路由是由管理员手工添加的路由条目&#xff1b;通过静态路由添加的都是非直连网段。 2.静态路由的特点&#xff1f; 静态路由的添加和删除都需要手工完成&#xff1b;静态路由无法适应网络的动态变更&#xff0c;即缺乏适应…

cisco packet tracer配置网络路由

广州大学 计算机网络实验 配置网络路由 利用packet tracer搭建如图网络 中间是三个路由器&#xff0c;两边各接一台计算机。 首先先把网络搭建出来 1是路由器&#xff0c;2是终端设备&#xff0c;3是连接设备的线缆。左键点击1或者2或3&#xff0c;区域4就会出现不同的路由器…

路由的几个基本概念-直连路由/网关路由/主机路由/网络路由/动态路由/静态路由/默认路由

1.动态路由/静态路由 1&#xff09;动态路由 路由选择器自动共享路由信息 自动构造路由表&#xff0c;需要一个路由协议&#xff0c;如RIP或OSPF 2&#xff09;静态路由 路由选择器不共享路由信息&#xff08;单方向路由&#xff09; 手工构造路由表 2.直连路由/网关路由…

计算机网络-实验四:配置网络路由

一、实验目的 了解路由器的特点、基本功能及配置方法&#xff1b;使用模拟软件Packet Tracer 8.0&#xff0c;熟悉Cisco路由器的操作&#xff1b;配置静态路由和距离矢量路由协议RIP&#xff0c;实现给定网络的连通&#xff1b;从而加深对IP编址、路由转发机制、路由协议、路由…

广州大学 计算机网络实验 2020版 配置网络路由

一、实验目的 了解路由器的特点、基本功能及配置方法&#xff1b;使用模拟软件 Packet Tracer 5.3 熟悉 Cisco 路由器的操 作&#xff1b;配置静态路由和距离矢量路由协议 RIP&#xff0c;实现给定网络的连通&#xff1b;从而加深对IP 编址、路由转发机制、路由协 议、路由表的…

Ad hoc网络路由协议概述1——分类

目录 1. 传统Internet网络路由协议 1.1 距离矢量路由协议&#xff08;Distance Vector&#xff09; 1.2 链路状态路由协议&#xff08;Link State&#xff09; 1.3 在Ad hoc网络中的不适用性 1.3.1 动态变化的网络拓扑结构 1.3.2 周期性的广播拓扑信息 1.3.3 单向的无线…

3.2 Ad Hoc 网络路由协议

Ad Hoc 网络路由协议 Ad Hoc 网络路由面临的问题 在设计Ad Hoc 网络路由协议时&#xff0c;我们首先要明确可能面临的问题&#xff1a; &#xff08;1&#xff09;路由信息不易获得&#xff1a;定期交换路由信息的开销大、网络资源有限&#xff0c;并且必须被所有节点共享、节…

Ad hoc网络路由协议概述2——表驱动路由协议(1)DSDV协议(Destination-sequenced distance vector protocol)

目录 1 DSDV协议的先导协议: DV协议的困境 2 解决DV协议计数到无穷的困境 2.1 毒性反转 (Poisoned reverse) 2.2 增加序列号 3 DSDV协议 3.1 基本流程 3.2 广播机制 3.2.1 交换路由信息的两个时刻 3.2.2 交换路由信息的两种方式 3.2.3 序列号不同如何选择 3.2.4 序列…

Ad hoc网络路由协议概述4——按需路由协议(2)AODV协议 (Ad-hoc on-demand distance vector algorithm protocol)

目录 1 一点前言 2 路由发现 2.1 相关概念 2.2 AODV的路由发现过程 2.3 与DSDV协议的对比 3 路由表管理及维护 3.1 更新路由表的策略 4 AODV协议的特点 4.1 优点 4.2 缺点 1 一点前言 在之前提过的DSR协议中&#xff0c;采用了源节点路由方式&#xff0c;每个数据报…

计算机网络实验四:配置网络路由

1、相关知识点 1.1 路由器的一般知识&#xff1a; 路由器是局域网与广域网之间进行互联的关键设备。通过它不仅可以互联不同协议、不 同物理接口的网络&#xff0c;还能选择数据传送的路经&#xff0c;并能阻隔非法访问。它在异构网互联能力、 拥塞控制能力和网段的隔离能力等方…

linux 默认路由 主机路由 网络路由

route命令 oute 命令的输出项说明 输出项 说明 Destination目标网段或者主机Gateway网关地址&#xff0c;”*” 表示目标是本主机所属的网络&#xff0c;不需要路由Genmask网络掩码Flags标记。一些可能的标记如下&#xff1a; U — 路由是活动的 H — 目标是一个主机 G — 路…

计算机网络路由转发题

1&#xff09;、目的地址和142.150.64.0/24明显不匹配&#xff0c;所以只有B、C、D&#xff0c;把相应的网络地址算出来 B-网络地址&#xff1a;142.150.71.128 C-网络地址&#xff1a;142.150.71.128 D-网络地址&#xff1a;142.150.0.0&#xff0c;把目的地址逐条的和子网…

各种路由的概念-直连路由、网关路由、主机路由、网络路由等

各种路由的概念 路由的分类 直连路由在添加的时候使用的是出接口&#xff08;dev&#xff09; 网关路由在添加的时候使用的是下一跳&#xff08;gw&#xff09; 主机路由的目的地址是一个完整的主机地址&#xff08;host&#xff09; 网络路由的目的地址是一个网络地址&#…