AcWing 1073. 树的中心(详解树形DP和换根DP)

news2025/1/16 20:02:07

AcWing 1073. 树的中心(树形DP + 换根DP)

  • 一、问题
  • 二、思路
    • 1、暴力做法
    • 2、树形DP+换根DP
      • (1)思路分析
      • (2)普通树形DP与换根DP的区别
  • 三、代码

一、问题

在这里插入图片描述

二、思路

1、暴力做法

这道题其实暴力的做法很简单,我们可以利用DFS去求出每个点到达树中其他点的最远距离。然后在这些最远距离中选出一个最大值。

暴力做法的代码如下:

#include<bits/stdc++.h>
using namespace std;
const int N = 1e4+10,M = 2*N,INF=0x3f3f3f3f;
int h[N],e[M],ne[M],w[M],idx;
int ans = INF;
void add(int x,int y,int z)
{
    e[idx]=y,w[idx]=z,ne[idx]=h[x],h[x]=idx++;
}
int dfs(int u,int father)
{
    int maxv=0;
    for(int i=h[u];i!=-1;i=ne[i])
    {
        int j=e[i];
        if(j==father)continue;
        maxv=max(maxv,dfs(j,u)+w[i]);
    }
    return maxv;
}
int main()
{
    memset(h,-1,sizeof h);
    int n;
    cin>>n;
    for(int i=0;i<n-1;i++)
    {
        int a,b,c;
        scanf("%d%d%d",&a,&b,&c);
        add(a,b,c),add(b,a,c);
    }
    for(int i=1;i<=n;i++)
    {
        ans=min(ans,dfs(i,-1));
    }
    cout<<ans<<endl;

    return 0;
}

那么我们分析一下这个做法是否可行。

我们想要知道一个树中的一个点到达树中其他点的最远距离的话,就需要去遍历树中的所有点,这个过程的时间复杂度就是 O ( n ) O(n) O(n),而我们需要对n个点都进行同样的操作,这样才能在这些最大的距离中挑出一个最小的。那么总的时间复杂度就是 O ( n 2 ) O(n^2) O(n2)。那么我们的运行次数最多就是 1 0 4 ∗ 1 0 4 = 1 0 8 10^4*10^4 = 10^8 104104=108这样做的效率就非常低了,很容易超时。

现实的确如此:
在这里插入图片描述

2、树形DP+换根DP

(1)思路分析

这道题在进行DP的分析之前需要大家会另外一道题,AcWing 1072. 树的最长路径(DFS与树形DP)

这道题就是在那道题的基础上做出的变形。

这道题的关键问题是如何高效地解决当前点到其他点的最远距离,我们可以将这些距离分个类,总共分为两类,一类是该节点沿着子树出发,到子树中某个节点的距离,另一类是从该节点的父节点出发,通过父节点到那些非该点的子树的点的距离。

即一个是向下出发的最大值,一个是向上出发最大值。这两个最大值之间的最大值就是我们想要求的。

可以用下面的图表示:

在这里插入图片描述
通过子树的最大值的求法,即向下出发的最大值就是我们刚才说的那道题中的求法,只需要将那道题中的dfs函数原封不动的移动过来即可。AcWing 1072. 树的最长路径(DFS与树形DP)

在这里主要讲解的是如何求向上出发的最大值。

在向下出发的最大值中,可能有很多情况,因为一个点可能会连接很多的子节点,但是向上出发的就不一样了。因为这是个树,而树中的子节点的父节点只有一个。因此,我们只需要考虑一种从父节点出发的情况。

我们以图中的B点为例子:

B点的父节点是A,那么B点的第二类距离可以描述为,B点到父节点A的距离加上A点到非B点的最大值。

那么对于A点而言,我们知道A点的距离有:A点向下出发的最大值和次大值,以及A点向上出发的最大值。

因此,我们需要在这三个值之间求出一个最大值再加上我们B到A的距离,就是B点向上出发的最大值。

这么想对吗?

我们考虑下面一种情况:

在这里插入图片描述
因为B点是A点的子节点,所以A点向下出发的距离中的最大值有可能是经过B点的,比如图中的绿线表示的情况,此时如果按照我们刚才的理解,最后的比较结果就有可能出现,A到B的距离加上A的向下出发的距离中的最大值,此时这个方式所描述的路线经过了2次B点。

这种情况是不合法的,也就是说此时我们只能在求解B点向上的最大值的时候,只能比较A点向下的次大值和A点向上的最大值。

相反如果A点向下的最大值不经过B点,那么我们只需要在A点向上的最大值中和A点向下的最大值中求出一个最大值。

说到这里,上面所说的两种情况就是我们接下来求解的思路了。为了方便两种情况之间的判断,我们还需要再开辟两个数组去记录向下出发距离中次大值和最大值是从哪个子节点转移过来的。

那么我们用 f [ 1 ] [ u ] f[1][u] f[1][u] f [ 2 ] [ u ] f[2][u] f[2][u]分别表示向下出发的次大值和最大值。

然后利用 g [ u ] g[u] g[u]表示向上出发的最大值。

s [ 1 ] [ u ] s[1][u] s[1][u] s [ 2 ] [ u ] s[2][u] s[2][u]分别表示u点向下出发的最大值经过子节点和向下出发的次大值经过的子节点。

状态转移方程就是我们上面描述的过程。

那么我们求向下的最大值是利用子节点更新父节点,而求向上的最大值是利用父节点更新子节点,前者就是普通的树形DP而后者则是树形DP的变形,叫做换根DP。

换根DP的代码的区别又在哪里呢?

我们看下面的讲解。

(2)普通树形DP与换根DP的区别

我们发现求第一类的时候,我们是用子节点去更新父节点,但是求第二类的距离的时候我们是用父节点更新子节点的。

那么这种利用父节点更新子节点的方式叫做:换根DP

那么普通的树形DP和换根DP是怎么体现的呢?

其实这就是一个先后更新的问题:

我们直接看下面的结果:

普通的树形DP

void dfs(int u,int father)
{
		for(遍历子节点)
		{
			dfs(子节点,u);
			dp[u]=xxxx;//更新dp数组
		}	
}

换根DP

void dfs(int u,int father)
{
		for(遍历子节点)
		{
			dp[u]=xxxx;//更新dp数组
			dfs(子节点,u);
		}	
}

这两个DP的唯一区别就是DFS和DP数组更新的顺序不同。

很好理解,换根DP是先更新父节点,再利用父节点去更新子节点,因此需要先执行dp数组更新的代码。普通的树形DP相反。

三、代码

#include<bits/stdc++.h>
using namespace std;
const int N = 1e4+10, M = 2 * N;
const int INF = 0x3f3f3f3f;
int h[N], e[M], ne[M], w[M], idx;
int f[4][N];
int s[2][N];
int g[N];

void add(int a, int b, int c)
{
    e[idx] = b, ne[idx] = h[a], w[idx] = c;
    h[a] = idx++;
}

void dfs_d(int u, int father)
{
    for(int i = h[u]; i != -1; i = ne[i])
    {
        int j = e[i];
        if(j == father)continue;
        dfs_d(j, u);
        if(f[1][j] + w[i] >= f[1][u])
        {
            f[2][u] = f[1][u];
            s[2][u] = s[1][u];
            f[1][u] = f[1][j] + w[i];
            s[1][u] = j;
        }
        else if(f[1][j] + w[i] >= f[2][u])
        {
            f[2][u] = f[1][j] + w[i];
            s[2][u] = j;
        }
    }
    return;
}

void dfs_u(int u ,int father)
{
    for(int i = h[u]; i != -1; i = ne[i])
    {
        int j = e[i];
        if(j == father)continue;
        if(j == s[1][u])
            g[j] = max(f[2][u], g[u]) + w[i];
        else
            g[j] = max(f[1][u], g[u]) + w[i];
        dfs_u(j,u);
    }
}

int main()
{
    memset(h, -1, sizeof h);
    int n;
    cin>>n;
    for(int i = 0; i < n - 1; i ++ )
    {
        int a, b, c;
        scanf("%d%d%d", &a, &b, &c);
        add(a, b, c), add(b, a, c);
    }
    dfs_d(1, -1);
    dfs_u(1, -1);
    int minv = INF;
    for(int i = 1; i <= n; i ++ )
    {
        minv = min(minv, max(g[i], f[1][i]));
    }
    cout<< minv << endl;
    return 0;
}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/187425.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【金融学】Economics of Money and Banking {暂时搁置,中级宏观和微观经济学未学}

Economics of Money and BankingClass1 The Big PicturePrerequisitesSome MaterialsCourse Material: https://www.coursera.org/learn/money-banking/lecture/8WXSW/the-big-picture Class1 The Big Picture Prerequisites intermediate macroeconomics 中级宏观经济学 int…

Java on VS Code 2023年1月更新|Spring 插件包、代码补全更新以及性能改进

作者&#xff1a;Nick Zhu - Senior Program Manager, Developer Division at Microsoft 排版&#xff1a;Alan Wang 大家好&#xff0c;欢迎来到我们 2023 年的第一篇博客&#xff01;我们想与您分享几个与 Spring 插件、代码编辑和性能相关的激动人心的更新&#xff0c;让我们…

XML方式—解决mybatis实体类属性名和数据库字段名不一致问题

数据库字段与类属性名称不一致&#xff0c;导致查询数据时数据没有封装上。 [Brand{id1, brandNamenull, companyNamenull}, Brand{id2, brandNamenull, companyNamenull}]解决方式一&#xff08;为表字段取别名&#xff09; <select id"selectAll" resultType&…

【大唐杯备考】——5G网络组网部署(学习笔记)

&#x1f4d6; 前言&#xff1a;本期介绍5G网络组网部署。 目录&#x1f552; 1. SA组网和NSA组网&#x1f558; 1.1 SA组网&#x1f558; 1.2 NSA组网&#x1f564; 1.2.1 Option 3系列&#x1f564; 1.2.2 Option 7系列&#x1f564; 1.2.3 Option 4系列&#x1f558; 1.3 组…

【Linux】gcc编译器

【Linux】gcc编译器 文章目录【Linux】gcc编译器1、背景2、翻译过程2.1 预处理2.2 编译2.3 汇编2.4 链接2.5 总过程3、动、静态链接库3.1 查看库3.2 库分类3.3 链接使用4、gcc参数总结1、背景 在Linux中&#xff0c;C语言的编译肯定需要编译器来进行&#xff0c;这个编译器就是…

微课录制软件哪个好用?微课制作方法分享

很多小伙伴都不知道什么是微课&#xff1f;微课视频时长比较短&#xff0c;一般在5-10分钟。微课的教学内容也比较少&#xff0c;一个微课视频一般只对一个知识点进行针对性讲解。微课的这些特性可以让我们利用碎片时间进行学习。制作微课需要录制电脑屏幕&#xff0c;那微课录…

Python 部分位运算(包含负数反码手算求法)

首先要回忆负数的补码的求法&#xff1a;写出其相反数的原码&#xff0c;而后从右往左&#xff0c;碰到第一个1后上面的数全取反。 下面以4位二进制的-5举例&#xff08;首位为符号位&#xff09;&#xff1a; 5的原码 0101末位遇1后取反 1011即-5的补码为 1011下面是部分pytho…

C/C++ 高精度(加减乘除)算法压位优化

文章目录前言一、基本原理1、存储方式2、计算方式二、完整代码三、性能对比总结附录1、性能测试代码前言 由于上一章《C/C 高精度&#xff08;加减乘除&#xff09;算法简单实现》实现了基本的高精度计算&#xff0c;数组的每个元素存储一位10进制的数字。这样的存储方式并不是…

数学建模学习笔记(15)时间序列分析

时间序列分析时间序列分析概述和数据预处理时间序列分解模型指数平滑模型ARIMA模型时间序列分析概述和数据预处理 时间序列的概念&#xff1a;也称为动态序列&#xff0c;是指将某种现象的指标值按照时间顺序排列而成的数值序列。 时间序列的组成要素&#xff1a;时间要素、数…

表格控件Aspose.Cells for Java 授权须知

Aspose API支持流行文件格式处理&#xff0c;并允许将各类文档导出或转换为固定布局文件格式和最常用的图像/多媒体格式。 Aspose.Cells for Java 是一个屡获殊荣的Excel电子表格处理API&#xff0c;它允许Java开发人员在自己的Java应用程序中嵌入可读取、写入和操作Excel电子…

天云数据Hubble数据库被评为“IDC创新者:中国分布式数据库,2022”

近日&#xff0c;国际知名研究机构 IDC 发布《IDC Innovator&#xff1a;中国分布式关系型数据库》报告正式发布&#xff0c;天云融创数据科技&#xff08;北京&#xff09;有限公司&#xff08;以下简称“天云数据”&#xff09;被评为“IDC创新者&#xff1a;中国分布式数据库…

经过ASEMI整流桥MB10F后输出电压是多少

编辑-Z 型号&#xff1a;MB10F 封装&#xff1a;MBF-4 最大重复峰值反向电压&#xff08;VRRM&#xff09;&#xff1a;1000V 最大平均正向整流输出电流&#xff08;IF&#xff09;&#xff1a;1.0A 峰值正向浪涌电流&#xff08;IFSM&#xff09;&#xff1a;35A 每个元…

清楚姐姐玩翻翻乐[期望dp]

首先这肯定是个期望dp。 首先明确二点 1.一旦这张牌已经知道数字了&#xff0c;下次翻他肯定是给他配对 2.如果已经知道了两张相同数字的卡片&#xff0c;那么在之后什么时候翻面都不会影响操作次数 故dp状态只需要记录只知道一张的卡片数量和未知的卡片数量。 接下来一个问题就…

多微网优化调度(风机、光伏、蓄电池、燃料电池、柴油机、电网交互)(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

【c语言进阶】文件操作(上)

&#x1f680;write in front&#x1f680; &#x1f4dc;所属专栏&#xff1a;> c语言学习 &#x1f6f0;️博客主页&#xff1a;睿睿的博客主页 &#x1f6f0;️代码仓库&#xff1a;&#x1f389;VS2022_C语言仓库 &#x1f3a1;您的点赞、关注、收藏、评论&#xff0c;是…

【C++11新特性】| 并发编程

文章目录一、std::thread1.1 成员函数1.2 案例二、std::mutex三、std::lock3.1 lock_guard3.2 unique_lock3.3 lock_guard与unique_lock的对比四、std::atomic五、volatile六、condition_variable成员函数七、future、promise、packaged_task7.1 std::promise7.2 std::future7.…

【Cloudcone】VPS 登录、防火墙设置

服务器登录、防火墙设置本来是一件很简单的事情&#xff0c;但是我买了国外服务器&#xff0c;有些地方出现了一些问题。 文章目录一、简介二、使用ipv6登录服务器三、防火墙设置四、小插曲一、简介 我一直用的腾讯云的服务器&#xff0c;但是带宽太低了&#xff0c;只有4M。最…

2023年二月份图形化三级打卡试题

活动时间 从2023年 2月1日至1月21日&#xff0c;每天一道编程题。 本次打卡的规则如下&#xff1a; &#xff08;1&#xff09;小朋友每天利用10~15分钟做一道编程题&#xff0c;遇到问题就来群内讨论&#xff0c;我来给大家答疑。 &#xff08;2&#xff09;小朋友做完题目后&…

CSS第四章——文本属性

文章目录4.1 文本颜色4.2 对齐文本4.3 装饰文本4.4 文本缩进4.5 行间距一些学习笔记。 CSS Text&#xff08;文本&#xff09;属性可定义文本的外观&#xff0c;比如文本的颜色、对齐文本、装饰文本、文本缩进、行间距等。 属性表示注意点color文本颜色通常使用十六进制 而且…

【算法题解】13. 删除链表的倒数第 N 个结点

文章目录题目解法一&#xff1a;计算链表长度Java 代码实现Go 代码实现复杂度分析解法二&#xff1a;双指针Java 代码实现Go 代码实现复杂度分析这是一道 中等难度 的题。 题目来自&#xff1a;https://leetcode.cn/problems/remove-nth-node-from-end-of-list/description/ 题…