首页 > 其他分享 >动态规划(六)——树形dp

动态规划(六)——树形dp

时间:2024-02-17 20:56:14浏览次数:28  
标签:子树 int 树形 为根 动态 节点 dp

树形dp,又称树状dp,即在树上进行的dp,在设计动态规划算法时,一般就以节点从深到浅(子树从小到大)的顺序作为dp的“阶段”,dp的状态表示中,第一维通常是节点编号(代表以该节点为根的子树)。大多数时候,我们采用递归的方式实现树形动态规划。对于每个节点x,先递归在他的每个子节点上进行dp,在回溯时从子节点向节点x,进行状态转移。

例题:

没有上司的舞会(Acwing 285)

题目描述
Ural 大学有 N 个职员,编号为 1~N。他们有从属关系,也就是说他们的关系就像一棵以校长为根的树,
父结点就是子结点的直接上司。每个职员有一个快乐指数。现在有个周年庆宴会,要求与会职员的快乐
指数最大。但是,没有职员愿和直接上司一起与会。

输入格式
第一行一个整数 N。(1<=N<=6000) 接下来 N 行,第 i+1 行表示 i 号职员的快乐指数 Ri (-128<=Ri<=127) 
接下来 N-1 行,每行输入一对整数 L,K。表示 K 是 L 的直接上司。 最后一行输入 0,0。

输出格式
输出最大的快乐指数。

样例
样例输入
7
1
1
1
1
1
1
1
1 3
2 3
6 4
7 4
4 5
3 5
0 0
样例输出
5

思路:

所有点形成一个森林,设f[v][0]表示v为根的子树如果v 不参加舞会,能得到的最大快乐指数,f[v][1]表示v为根的子树如果v参加舞会,能得到的最大快乐指数。那么在计算f[v]的时候,可以先递归计算其所有孩子的f值,然后考虑f[v],对于f[v][0],由于v没参加,所以其子树树根可以参加也可以不参加,所以
f [v] [0]=∑ v → u max ⁡{f [u][0],f[u][1]} 
对于f[v][1],由于v参加了,所以其所有子树树根都不能参加,所以
f [v][1]=∑ v → u f[u][0]
最后对于所有树根vi ,求一下∑ i max ⁡{f[vi][0],f [vi] [1]} 即为答案。

#include <bits/stdc++.h>
#define N 10010
using namespace std;
vector<int> son[N];
int f[N][2],v[N],h[N],n;
void dp(int x){
	f[x][0]=0;
	f[x][1]=h[x];
	for(int i=0;i<son[x].size();i++){
		int y=son[x][i];
		dp(y);
		f[x][0]+=max(f[y][0],f[y][1]);
		f[x][1]+=f[y][0];
	}
}
int main(){
	cin>>n;
	for(int i=1;i<=n;i++) cin>>h[i];
	for(int i=1;i<n;i++){
		int x,y;
		cin>>x>>y;
		v[x]=1;
		son[y].push_back(x);
	}
	int root;
	for(int i=1;i<=n;i++)
	    if(!v[i]){
	    	root=i;
	    	break;
		}
		dp(root);
		cout<<max(f[root][0],f[root][1])<<endl;
}

#一名爱玩狂铁的oier#

标签:子树,int,树形,为根,动态,节点,dp
From: https://www.cnblogs.com/hzoiwzs/p/18018381

相关文章

  • 回顾复习之坐标DP
    定义坐标型动态规划一般是给定网格、序列,求满足条件的MAX或MIN。开数组时,dp[i]一般代表以ai结尾的满足条件的子序列,dp[i][j]代表以i、j结尾的满足条件的最优解例题数塔典中典变形晴天小猪历险记之Hill抓苹果免费馅饼矩阵取数描述传送门思路首先看出,每行的问题是独立......
  • 动态规划(五)——坐标dp
    传纸条题目描述小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了。幸运的是,他们可以通过传纸条来进行交流。纸条要经由许多同学传到对方手里,小......
  • DP总结
    DP总结DP(动态规划)简介动态规划是一种通过把原问题分解为相对简单的子问题的方式求解复杂问题的方法。由于动态规划并**不是某种具体的算法**,而是一种解决特定问题的方法,因此它会出现在各式各样的数据结构中,与之相关的题目种类也更为繁杂。DP基础1.必要前提 需要满足三个......
  • 动态规划目录
    一.背包DP     1.0/1背包     2.完全背包     3.多重背包     4.分组背包二.线性DP三.背包DP......
  • 区间dp
    Ø合并:意思就是将两个或多个部分进行整合,当然也可以反过来,也就是是将一个问题进行分解成两个或多个部分。Ø特征:能将问题分解成为两两合并的形式Ø求解:对整个问题设最优值,枚举合并点,将问题分解成为左右两个部分,最后将左右两个部分的最优值进行合并得到原问题的最优值。有点类......
  • 区间dp
    合并类动态规划合并:意思就是将两个或多个部分进行整合,当然也可以反过来,也就是是将一个问题进行分解成两个或多个部分。特征:能将问题分解成为两两合并的形式求解:对整个问题设最优值,枚举合并点,将问题分解成为左右两个部分,最后将左右两个部分的最优值进行合并得到原问题的最优......
  • 区间dp
    1.合并石子(1)排成一列的石子这个与合并果子唯一的不同就是只能合并相邻的石子,所以贪不得(怎么所有类型的动规要先上来搞贪心,有点diss贪心的感觉)f[i][j]表示i到j间合并的最大/最小得分;核心for(intlen=2;len<=n;len++){//表示长度2到len时的最大 for(inti=1;i+len-1<=n;i++)......
  • 背包dp
    01背包描述:有n个物品,每个物品只有一件,第i个物品体积为vi,价格为ci。现在有一个体积为V的背包,请你从n件物品里选出若干件放进背包里,使得背包里的物品价值最大。思路:01背包的特点是:每种物品只有一件,可以选择放或不放。我们可以根据此特点进行动态规划(DP),设f[i][j]表示前i件物品放......
  • 动态规划-DP 完整版
    动态规划学完了五大基础dp做个简单总结dp特征动态规划问题首要是找到做题的目的是求最大/小值还是其他;其二要确定问题的状态转移方程这是关键;第三为dp数组找到边界、最后检查是否需要结合其他相关知识如树dfs等;别忘了检查多测输入数组变量置零等易错点。......
  • 线性dp
    线性动态规划:不用多说,主要应用于求上升子序列,下降子序列等直接看例题:样例输入:13791638243718441921226315样例输出:max=879161819212263解:#include<bits/stdc++.h>usingnamespacestd;constintMAX=1050;intn,ans;intf[MAX],......