算法:树形dp(树状dp)

文章目录

    • 一、树形DP的概念
      • 1.基本概念
      • 2.解题步骤
      • 3.树形DP数据结构
    • 二、典型例题
      • 1.LeetCode:337. 打家劫舍 III
        • 1.1、定义状态转移方程
        • 1.2、参考代码
      • 2.ACWing:285. 没有上司的舞会
        • 1.1、定义状态转移方程
        • 1.2、拓扑排序参考代码
        • 1.3、dfs后序遍历参考代码

一、树形DP的概念

  树形动态规划(Tree DP) 是动态规划在树结构数据上的应用。树是一种特殊的图,具有无环的特点,这使得树形动态规划在很多问题中比普通的动态规划更为直观和高效。在树形DP问题中,通常需要处理与节点相关的状态,并利用树的层次结构,自底向上或自顶向下递归地解决问题。树形DP的学习对线性DP的状态定义也有很大帮助。

1.基本概念

  • 状态表示:树形DP的状态通常与树的节点相关联,每个状态代表了以某节点为根的子树在某种条件下的最优解、计数或其他性质。
  • 状态转移:状态的转移依赖于子节点到父节点(或父节点到子节点)的关系。通常,要解决一个节点的问题,需要先解决它的所有子节点的问题,然后根据子节点的解来更新当前节点的解。或者当前结点的状态由父亲转移而来。

  在树形DP中,如果当前结点的状态由其子节点转移而来,那么可以进行DFS后序遍历,或者拓扑排序顺序遍历(在图中)。比如在关键路径中,提到过,求最长路径可以使用拓扑排序+动态规划来求解,这实际上就很像是一个树形DP。

2.解题步骤

  我们说树形DP的状态转移,实际上比线性DP更直观,因为父子结点之间的关系是很明显的,是从上到下的状态转移还是从下依次往下的状态转移是很明显的。

  1. 定义状态:首先明确每个状态所代表的含义,以及解题需要考虑的所有情况。
  2. 状态转移方程:根据问题的具体条件,找出不同状态之间的关系,形成状态转移方程。这通常涉及对当前节点的处理和对子节点状态的汇总。
  3. 初始化:确定递归的基本情况,即最简单情况下各状态的值,用于终止递归。
  4. 计算顺序:确定状态计算的顺序。在树形DP中,通常需要后序遍历(先处理所有子节点,再处理父节点)来自底向上计算,或者先序遍历来自顶向下传递信息。
  5. 答案提取:根据定义的状态,从根节点或特定节点提取最终答案。

3.树形DP数据结构

线性dp直接可以使用线性数组来存储状态,那么树形DP怎么办呢?

  • 如果真的只跟其亲儿子有关,你可以直接在树上进行DFS后序,利用DFS函数中存储状态,因为这不涉及到记忆化搜索
  • 如果其不仅仅跟其亲儿子有关,你就必须采取一定行动来保存状态信息了(属于一种记忆化搜索)。
  • 你可以先进行一次DFS对树结点编号(或已经有编号),然后再操作。这样你就可以定义dp数组了
  • 你可以对树结点存储在哈希表中来存储该结点状态。这样你就可以定义dp了
  • 如果需要新建图,你可以对树或图,存储其邻接表,并且每个结点多定义一个状态信息即可。若对于一个特定的顶点状态,它跟其邻接表中所有结点有关,如果其邻接表中的顶点状态已经被求出,则可以进行转移。树图不分家。vector实现邻接表
  • ···

二、典型例题

1.LeetCode:337. 打家劫舍 III

模板题
337. 打家劫舍 III
在这里插入图片描述
  父亲最优解跟其子节点最优解有关,因此属于树状dp。由于父亲的最优解不仅跟其亲儿子有关,还跟孙子有关,因此我们在进行树状dp时,不能只DFS后序遍历返回结果,我们还得存储孙子信息,即记忆化搜索,因此我们在书写这个树状dp的时候,需要使用额外空间存储状态,比如在此题中没有给出编号,使用哈希表是最快的。
  但是以上我们所说的状态是单个状态 即 dp[root]表示以root为根的最高金额,如果我们定义的状态dp[root][0]表示不选择root时的最高金额,dp[root][1]表示选择root时的最高金额,那么我们只需要将树返回值是多个值(如结构体),就可以不用多开额外空间了~

1.1、定义状态转移方程
  • dp[root][0]=max(dp[l_child][1],dp[l_child][0])+max(dp[r_child][1],dp[r_child][0]);//不被选择时
  • dp[root][1]=root->val+dp[l_child][0]+dp[r_child][0];//被选择时
    • 根不被选择时,其能得到的最大金额的状态,由其儿子不被选择的最大金额状态以及其儿子被选择的最大金额状态转移而来,因为根不被选择时,其儿子可以被选也可以不被选,让根的金额最大,那么择其最大者就行。
    • 根被选择时,其能得到的最大金额的状态,由其儿子不被选择的最大金额状态转移而来。
1.2、参考代码
class Solution {
public:
	//pair::first表示不被选择时的最大值
	//pair::second表示被选择时的最大值
    pair<int,int> TreeDp(TreeNode * root){
        if(!root) return {0,0};
        pair<int,int> dp_left=TreeDp(root->left);
        pair<int,int> dp_right=TreeDp(root->right);
        pair<int,int> dp_root;
        //不被选择时
        dp_root.first=max(dp_left.first,dp_left.second)+max(dp_right.first,dp_right.second);
        //被选择时
        dp_root.second=root->val+dp_left.first+dp_right.first;
        return dp_root;
    }
    int rob(TreeNode* root) {
        pair<int,int> ans=TreeDp(root);
        return max(ans.first,ans.second);
    }
};

2.ACWing:285. 没有上司的舞会

模板题
285. 没有上司的舞会
在这里插入图片描述
  这题和 打家劫舍 III 基本上一摸一样呀,只不过这里可能是图,并且acwing需要自己构建结点,因此这里可以提供更多建树dp思路。

没有职员愿意和直接上司一起参会。实际上就是指相邻父子结点不能同时被选择。

1.1、定义状态转移方程

  这里我们的状态和打家劫舍III的状态一模一样。不过我们可以使用拓扑排序的方式实现,只需要存储fa数组,表示父节点,因为根结点的状态由子节点转移而来,并且没必要按顺序转移~。用邻接表存图,然后按照打家劫舍III dfs实现会简单很多。

1.2、拓扑排序参考代码
#include<bits/stdc++.h>
using namespace  std;
int main(void){
    ios_base::sync_with_stdio(false);
    cin.tie(0);
    int N;
    cin>>N;
    vector<int> fa(N+1,-1);//拓扑排序方法 找寻前驱用
    vector<int> degree(N+1,0);//用于拓扑排序删除度的数组
    vector<pair<int,int>> dp(N+1);//dp状态数组,second表示被选择,first表示不被选择。
    for(int i=1;i<=N;++i) {cin>>dp[i].second;dp[i].first=0;}
    for(int i=1;i<N;++i){
        int son,Fa;
        cin>>son>>Fa;
        fa[son]=Fa;
        degree[Fa]++;
    }
    //为了和打家劫舍不一样,我们这里采用拓扑排序的方式
    //如果仍然用dfs,则需要找到根结点,dfs会方便很多,也不需要fa数组了
    stack<int> sta;
    for(int i=1;i<=N;++i){
        if(degree[i]==0){
            sta.push(i);
        }
    }
    int ans=0;
    while(!sta.empty()){
        int cur=sta.top();sta.pop();
        if(fa[cur]!=-1) {
            dp[fa[cur]].first += max(dp[cur].first, dp[cur].second);
            dp[fa[cur]].second += dp[cur].first;
            if (--degree[fa[cur]] == 0) sta.push(fa[cur]);
        }else
            ans=max(ans,max(dp[cur].first,dp[cur].second));
    }
    cout<<ans;
    return 0;
}
1.3、dfs后序遍历参考代码
#include<bits/stdc++.h>
using namespace  std;
vector<vector<int> > g;
vector<pair<int,int>> dp;//dp状态数组,second表示被选择,first表示不被选择。
void dfs(int root){
	for(int i=0;i<g[root].size();++i){
		dfs(g[root][i]);
		dp[root].second+=dp[g[root][i]].first;
		dp[root].first+=max(dp[g[root][i]].first,dp[g[root][i]].second);
	}
	return;
}
int main(void){
    ios_base::sync_with_stdio(false);
    cin.tie(0);
    int N;
    cin>>N;
    g.assign(N+1,vector<int>{});
    dp.assign(N+1,{});//dp状态数组,second表示被选择,first表示不被选择。
    unordered_set<int> roots;//用于保存根
    for(int i=1;i<=N;++i) {cin>>dp[i].second;dp[i].first=0;roots.insert(i);}
    for(int i=1;i<N;++i){
        int son,Fa;
        cin>>son>>Fa;
        g[Fa].emplace_back(son);
        roots.erase(son);
    }
    int ans=0;
    for(auto & i:roots){
		dfs(i);ans+=max(dp[i].first,dp[i].second);//从不同根遍历
	}
    cout<<ans;
    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

MySQL复制拓扑4

文章目录 主要内容一.启用GUID并配置循环复制1.其中&#xff0c;UUID用来唯一标识每一个服务器&#xff0c;事务的编号记录了在该服务器上执行的事务的顺序。使用SELECT server_uuid\G命令可以查看服务器的UUID&#xff0c;sever1的UUID值显示如下&#xff1a;代码如下&#xf…

Vue3_2024_7天【回顾上篇watch常见的后两种场景】

随笔&#xff1a;这年头工作不好找咯&#xff0c;大家有学历提升的赶快了&#xff0c;还有外出人多注意身体&#xff0c;没错我在深圳这边阳了&#xff0c;真的绝啊&#xff0c;最尴尬的还给朋友传染了&#xff01;&#xff01;&#xff01; 之前三种的监听情况&#xff0c;监听…

文本识别 OCR 解决方案

Capture2Text 便携式 OCR 工具 Capture2Text 能够使用键盘快捷键快速对屏幕的一部分进行 OCR。 默认情况下&#xff0c;生成的文本将保存到剪贴板。支持中文、英文、法文、德文、日文、韩文、俄文、西班牙文等 90 多种语言。 Capture2Text 是便携式工具&#xff0c;不需要安装…

快速了解FastAPI与Uvicorn是什么?

概念 什么是Uvicorn Python Uvicorn 是一个快速的 ASGI&#xff08;Asynchronous Server Gateway Interface&#xff09;服务器&#xff0c;用于构建异步 Web 服务。它基于 asyncio 库&#xff0c;支持高性能的异步请求处理&#xff0c;适用于各种类型的 Web 应用程序。 Uvi…

SEO超级外链工具源码

源码简介 超级外链工具&#xff0c;是一款在线全自动化发外链的推广工具。使用本工具可免费为网站在线批量增加外链&#xff0c;大大提高外链发布工作效率&#xff0c;是广大草根站长们必备的站长工具。 搭建环境 PHP 5.6 安装教程 上传源码压缩包到网站目录并解压即可 首…

Linux安装最新版Docker完整教程

参考官网地址&#xff1a;Install Docker Engine on CentOS | Docker Docs 一、安装前准备工作 1.1 查看服务器系统版本以及内核版本 cat /etc/redhat-release1.2 查看服务器内核版本 uname -r这里我们使用的是CentOS 7.6 系统&#xff0c;内核版本为3.10 1.3 安装依赖包 …

【数据结构(二)】顺序表与ArrayList

❣博主主页: 33的博客❣ ▶文章专栏分类:数据结构◀ &#x1f69a;我的代码仓库: 33的代码仓库&#x1f69a; &#x1faf5;&#x1faf5;&#x1faf5;关注我带你学更多数据结构知识 目录 1.前言2.定义IList接口3.MyArraylist实现接口3.1定义成员变量与构造方法3.2添加元素3.3…

构建未来数字化世界的统一用户中心产品架构

随着数字化时代的到来&#xff0c;用户数据管理变得愈发复杂&#xff0c;各类应用和服务的涌现使得用户信息分散存储&#xff0c;导致了数据孤岛和体验碎片化的问题。在这样的背景下&#xff0c;统一用户中心产品架构应运而生&#xff0c;为构建数字化世界提供了全新的解决方案…

S7-200 SMART 应用第003期-数字量输入模块接线

概述 S7-200 SMART作为西门子的一款高性价比PLC产品,很多工控电气工程师在选型和电路图设计时,对模块接线并不是非常清楚,为了使大家更好的了解和掌握该部分,本文从CPU本体、数字量输入(DI)、数字量输出(DQ)向大家详细介绍S7-200 SMART 详细的接线和注意事项。 不同型号C…

2023年度总结:允许迷茫,破除迷茫;专注自身,把握当下

0、前言 &#x1f4dc;为什么24年已经过了几个月&#xff0c;才提笔写这年度总结呢&#xff1f;毫不羞愧直问我的内心&#xff0c;其实就是懒罢了。直到前几天朋友看到了我去年写的总结&#xff0c;我自己点进那篇总结&#xff0c;完完整整的看了一遍&#xff0c;又翻看我23年…

ideaSSM 网上选课管理系统bootstrap开发mysql数据库web结构java编程计算机网页源码maven项目

一、源码特点 idea 开发 SSM 网上选课管理系统是一套完善的信息管理系统&#xff0c;结合SSM框架和bootstrap完成本系统&#xff0c;对理解JSP java编程开发语言有帮助系统采用SSM框架&#xff08;MVC模式开发&#xff09;&#xff0c;系统具有完整的源代码和数据库&#xff…

二分法题集1

1 二分查找 分析&#xff1a; 这是一道很简单的二分法题&#xff0c;定义两个指针和中间值middle&#xff0c;判断middle对应数组值与目标值的大小关系&#xff0c;从而对left和right进行修改。由于太过基础&#xff0c;代码简单基础就不多赘述。 目录 1 二分查找 分析&…

PyQt PySide6零基础入门与项目实战视频教程

目录 课程亮点课程大纲第一章&#xff1a;基础篇 PySide6开发环境安装第二章 控件与布局篇 PySide6常用控件与界面布局使用介绍第三章 信号槽与事件机制第四章 QMainWindow应用篇第五章 样式表qss与自定义控件第六章 图表与曲线第七章 数据库编程第八章 项目实战&#xff1a;高…

FJSP:小龙虾优化算法(Crayfsh optimization algorithm,COA)求解柔性作业车间调度问题(FJSP),提供MATLAB代码

一、柔性作业车间调度问题 柔性作业车间调度问题&#xff08;Flexible Job Shop Scheduling Problem&#xff0c;FJSP&#xff09;&#xff0c;是一种经典的组合优化问题。在FJSP问题中&#xff0c;有多个作业需要在多个机器上进行加工&#xff0c;每个作业由一系列工序组成&a…

二叉树的介绍

学习堆排序时先了解下二叉树&#xff0c;因为堆排序中使用了二叉树。 一、二叉树介绍 二叉树&#xff08;binary tree&#xff09;树的每个节点最多有2个孩子节点。注意&#xff0c;这里是最多有2个&#xff0c;也可能只有1个&#xff0c;或者没有孩子节点。 二叉树结构如图…

极客时间: 用 Word2Vec, LangChain, Gemma 模拟全本地检索增强生成(RAG)

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

「 典型安全漏洞系列 」11.身份验证漏洞详解

身份验证是验证用户或客户端身份的过程。网站可能会暴露给任何连接到互联网的人。这使得健壮的身份验证机制成为有效的网络安全不可或缺的一部分。 1. 什么是身份验证 身份验证即认证&#xff0c;是验证给定用户或客户端身份的过程。身份验证漏洞使攻击者能够访问敏感数据和功…

RobotFramework测试框架(12)--第三方库

Library 关于射频指南 |机器人框架 (robotframework.org) 使用RF需要使用Library&#xff0c;常用的第三方库如下&#xff1a; 在web浏览器中进行web应用程序测试可以使用的库是 Selenium Library 在内部使用流行的 Selenium 工具的 Web 测试库Browser Library 由 Playwri…

ThingsBoard通过MQTT发送遥测数据

MQTT基础 客户端 MQTT连接 遥测上传API 案例 MQTT基础 MQTT是一种轻量级的发布-订阅消息传递协议&#xff0c;它可能最适合各种物联网设备。 你可以在此处找到有关MQTT的更多信息&#xff0c;ThingsBoard服务器支持QoS级别0&#xff08;最多一次&#xff09;和QoS级别1&…

【前沿模型解析】潜在扩散模 1 | LDM第一阶段-感知图像压缩总览

文章目录 0 开始~1 感知压缩的目的2 自回归编码器-解码器生成模型一览2.1 AE 自编码器2.2 VAE 变分自编码器2.3 VQ-VAE2.4 VQ-GAN 3 代码部分讲解总览 0 开始~ 从今天起呢&#xff0c;我们会剖析LDM&#xff08;潜在扩散模型&#xff09; 从去年开始&#xff0c;大量的生成模…