树形DP【初级版】
START:
2021-08-14
10:00:37
在树形DP中,我们可以用数据模拟出一张图,一般是一棵树或是森林,所有的节点一般最多只有一个父节点。并且树里面没有重边或者环,
因此,一颗有N个节点的树有N-1条边。
讲这么多有点抽象,我们借用树形DP最经典的入门题来介绍树形DP的操作:
洛谷P1352 没有上司的舞会
题目链接:
https://www.luogu.com.cn/problem/P1352
题目详情:
某大学有 n 个职员,编号为 1…n。
他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司。
现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数 ri,但是呢,如果某个职员的直接上司来参加舞会了,
那么这个职员就无论如何也不肯来参加舞会了。
所以,请你编程计算,邀请哪些职员可以使快乐指数最大,求最大的快乐指数。
输入格式
输入的第一行是一个整数 n。
第 2 到第 (n+1) 行,每行一个整数,第 (i+1) 行的整数表示 i 号职员的快乐指数 ri。
第 (n+2) 到第 2n 行,每行输入一对整数 l,k,代表 k 是 l 的直接上司。
输出格式
输出一行一个整数代表最大的快乐指数。
输入输出样例
7
1
1
1
1
1
1
1
1 3
2 3
6 4
7 4
4 5
3 5
5
说明/提示
数据规模与约定
对于 100% 的数据,保证 1≤n≤6×10^3,−128≤ri≤127,1≤l,k≤n,且给出的关系一定是一棵树。
建图:节点从1开始,自上层节点是下层节点的直属上司
如图所示 ,5是3,4的直属上司,3是1,2的直属上司,4是6,7的直属上司,每个人的快乐指数都是1。
这就是树形DP的最经典的模型。
分析:
我们看5号节点,5号节点有两种选法,一种是选,一种是不选,分别对应两种情况:
1.选:如果选择5号节点,那么他的直系下属3,4就不能选。
2.不选:如果不选择5号节点,那么他的直系下属可以选也可以不选。
我们定义dp[N][2]数组:
dp[N][0]表示在以第N号节点为根的树、在不选择第N号点的情况下的、这棵树的最大快乐指数。
dp[N][1]表示在以第N号及单为根的树、在选择第N号点的情况下的、这棵树的最大快乐指数。
所以我们最后需要求得的答案就是max(dp[5][0],dp[5][1])。
(PS:这里的5是本题所给的样例的最终根节点)
我们在回过头来看任意的根节点 i 与它的子节点的快乐指数的关系:
我们还是先以题目给的样例的5号节点来看,5号节点的子节点有3,4,那么:
对于dp[5][1],也就是选择5号节点,那么3,4号节点都不能选择,所以dp[5][1]+=dp[3][0]+dp[4][0]
对于dp[5][0],也就是不选择5号节点,那么3,4号节点可选可不选,
所以取最大值:dp[5][0]+=max(dp[3][0],dp[3][1])+max(dp[4][1],dp[4][0])
所以DP问题最核心的内容——状态转移方程就推出来了,我们用u表示当前的根节点,用v表示根节点u的所有子节点,所以:
这样每一层的转移方程就确定下来了,但是,现在有一个问题,我们是从最终根节点(样例中是5)开始更新每个节点的最大快乐指数的,但是我们更新上一层需要下一层的快乐指数,也就是说,我们需要先将下一层的点更新,我们怎么实现这一步呢?
其实我认为树形DP的实现过程是回溯的过程,也就是说树形DP实现的过程是从最下面的末端的点开始、自下而上更新完所有的点的,所以我们在写函数的时候需要递归,递归至末端截止,然后开始更新末端的点,然后回溯到上一个点,再更新上一个点,再回溯……直至最终根节点。
好!我们来实现看看吧:
我们先处理输入的数据,我们使用链接表来储存:
在添加边的函数add(int a,int b)中,a是父节点,b是子节点
在预处理函数init()中,我们得先将所有的节点的指向都改为-1,然后再输入下属和上司的编号,再从上司到下属之间连一条线,再用st[N]数组标记子节点b已经有父节点了,表示b节点不是最终根节点,st[b]=true
#include<iostream>
#include<cstring>
using namespace std;
const int N=6005;
int n;
int h[N],ne[N],e[N],idx;
int happy[N];
int dp[N][2];
bool st[N]; void add(int a,int b){
e[idx]=b,ne[idx]=h[a],h[a]=idx++;
} void init(){
memset(h,-1,sizeof h);
for(int i=1;i<=n;i++)scanf("%d",&happy[i]);
for(int i=0;i<n-1;i++){
int a,b;
scanf("%d%d",&a,&b);
add(b,a);
st[a]=true;
}
} void dfs(int root){ } int main()
{
cin>>n;
init();
return 0;
}
预处理完毕后,我们来处理核心函数dfs(int u),
我们先在dfs()中预处理dp[u][1]和dp[u][0] (dp[u][0]就是0,不需要预处理)
void dfs(int u){
dp[u][1]=happy[u]; }
然后遍历根节点u的所有子节点,
void dfs(int u){
dp[u][1]=happy[u];
for(int i=h[u];i!=-1;i=ne[i]){
int j=e[i]; }
}
我们在更新dp[u][1]和dp[u][0]的时候需要用到子节点的最大快乐指数,所以我们在更新dp[u][1]和dp[u][0]之前需要再dfs(u的子节点),来将子节点的最大快乐指数更新。然后再更新当前根节点u的最大快乐指数。
补充完整:
void dfs(int u){
dp[u][1]=happy[u];
for(int i=h[u];i!=-1;i=ne[i]){
int j=e[i];
dfs(j);
dp[u][1]+=dp[j][0];
dp[u][0]+=max(dp[j][1],dp[j][0]);
}
}
我们在main函数里面还要求出最终根节点root,先初始化为1,我们之前用st[N]数组标记过,如果st[i]==false,那么点i就是最终根节点,所以在main函数中,我们应该这样写:
int main()
{
cin>>n;
init();
int root=1;
while(st[root])root++;
dfs(root);
printf("%d\n",max(dp[root][1],dp[root][0]));
return 0;
}
所以完整代码如下:
#include<iostream>
#include<cstring>
using namespace std;
const int N=6005;
int n;
int h[N],ne[N],e[N],idx;
int happy[N];
int dp[N][2];
bool st[N]; void add(int a,int b){
e[idx]=b,ne[idx]=h[a],h[a]=idx++;
} void init(){
memset(h,-1,sizeof h);
for(int i=1;i<=n;i++)scanf("%d",&happy[i]);
for(int i=0;i<n-1;i++){
int a,b;
scanf("%d%d",&a,&b);
add(b,a);
st[a]=true;
}
} void dfs(int u){
dp[u][1]=happy[u];
for(int i=h[u];i!=-1;i=ne[i]){
int j=e[i];
dfs(j);
dp[u][1]+=dp[j][0];
dp[u][0]+=max(dp[j][1],dp[j][0]);
}
} int main()
{
cin>>n;
init();
int root=1;
while(st[root])root++;
dfs(root);
printf("%d\n",max(dp[root][1],dp[root][0]));
return 0;
}
树形DP【初级版】的更多相关文章
- 基础DP(初级版)
本文主要内容为基础DP,内容来源为<算法导论>,总结不易,转载请注明出处. 后续会更新出kuanbin关于基础DP的题目...... 动态规划: 动态规划用于子问题重叠的情况,即不同的子问 ...
- P1352 没有上司的舞会——树形DP入门
P1352 没有上司的舞会 题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司.现在有个周年庆宴会,宴会每邀请来一个职员 ...
- HDU 5977 Garden of Eden (树形dp+快速沃尔什变换FWT)
CGZ大佬提醒我,我要是再不更博客可就连一月一更的频率也没有了... emmm,正好做了一道有点意思的题,就拿出来充数吧=.= 题意 一棵树,有 $ n (n\leq50000) $ 个节点,每个点都 ...
- 树形DP学习笔记
树形DP 入门模板题 poj P2342 大意就是一群职员之间有上下级关系,每个职员有一个快乐值,但是只有在他的直接上级不在场的情况下才会快乐.求举行一场聚会的快乐值之和的最大值. 求解 声明一个数组 ...
- 树形DP初探•总结
这几天,我自学了基础的树形DP,在此给大家分享一下我的心得. 首先,树形DP这种题主要就是解决有明确分层次且无环的树上动态规划的题.这种题型一般(注意只是基础.普通的情况下)用深度优先搜索来解决实 ...
- UVA - 1218 Perfect Service(树形dp)
题目链接:id=36043">UVA - 1218 Perfect Service 题意 有n台电脑.互相以无根树的方式连接,现要将当中一部分电脑作为server,且要求每台电脑必须连 ...
- 树的直径的求法即相关证明【树形DP || DFS】
学习大佬:树的直径求法及证明 树的直径 定义: 一棵树的直径就是这棵树上存在的最长路径. 给定一棵树,树中每条边都有一个权值,树中两点之间的距离定义为连接两点的路径边权之和.树中最远的两个节点之间的距 ...
- UVa12186:Another Crisis(树形DP)
一道简单的树形DP送给你. A couple of years ago, a new world wide crisis started, leaving many people with econo ...
- [BZOJ 4455] [ZJOI 2016] 小星星 (树形dp+容斥原理+状态压缩)
[BZOJ 4455] [ZJOI 2016] 小星星 (树形dp+容斥原理+状态压缩) 题面 给出一棵树和一个图,点数均为n,问有多少种方法把树的节点标号,使得对于树上的任意两个节点u,v,若树上u ...
- hdu 6035:Colorful Tree (2017 多校第一场 1003) 【树形dp】
题目链接 单独考虑每一种颜色,答案就是对于每种颜色至少经过一次这种的路径条数之和.反过来思考只需要求有多少条路径没有经过这种颜色即可. 具体实现过程比较复杂,很神奇的一个树形dp,下面给出一个含较详细 ...
随机推荐
- make编译工具教程
make编译工具教程 背景 CC++编译起来目录结构多的情况需要脚本完成搜索编译-> make 第一课 第三课 makefile常用编程语法: 1 pipe管道符号,用;把命令相连接. 这样就 ...
- csp2020——T3表达式
后缀表达式基本可以使用栈来表达,所以30分的暴力做法很好做 正解的做法是: 暴力的做法是每次重新建立栈,用符号来把栈顶的元素弹出来,做完运算之后再放入栈中,如果是与运算,弹出两个元素,如果是或运算,弹 ...
- iOS笔记 - runtime 02:objc_msgSend执行流程
objc_msgSend 执行流程 1 - 第一步:消息发送 2 - 第二步:动态解析 代码示例:resolveInstanceMethod | resolveClassMethod 存在问题:68 ...
- (K8s学习笔记五)Pod的使用详解
1.Pod用法 K8s里使用的容器不能使用启动命令是后台执行程序,如:nohup ./start.sh &,该脚本运行完成后kubelet会认为该Pod执行结束,将立刻销毁该Pod,如果该Po ...
- luna lunatic
Luna是罗马神话的月神.英语中Lunacy.Lunatic等意指疯狂的字语源均来自Luna.月亮的阴晴圆缺影响地球的潮汐涨退甚至生物周期,故此古时的人们相信月亮拥有使人疯狂的魔力,人狼等传说亦是因此 ...
- docker镜像的获取、查看、删除、docker镜像管理、docker容器管理
在不想弄乱本地及其环境下该如何进行系软件的安装? 下载安装docker工具 获取该软件的docker镜像(你以后想要用各种工具,基本都能够搜索docker search nginx:版本号到合适的镜像 ...
- darkriscv笔记
1 按照默认设置,前4K空间为ROM,后4K空间是RAM.为什么RAM需要初值,初值从哪儿来? 2 指令分类 LUI: AUTPC: JAL: JALR: BCC : BEQ/BGE /BGEU/BL ...
- Echarts中国地图下钻
//各省份的地图json文件 var provinces = { '上海': '/asset/get/s/data-1482909900836-H1BC_1WHg.json', '河北': '/ass ...
- uniapp 通用函数说明
onLoad函数 监听页面加载,在onLoad中发送请求是比较合适的,即页面一加载就发送请求获取数据,option接受其他界面传过来的数据,数据类型为obj onLoad(option) { ...
- vue 切换json语言包
1.在asstes文件夹新建 2.在main.js读取本地json语言包 3.在vue页面t这样引用