P1352 没有上司的舞会——树形DP入门
P1352 没有上司的舞会
题目描述
某大学有N个职员,编号为1~N。他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司。现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数Ri,但是呢,如果某个职员的上司来参加舞会了,那么这个职员就无论如何也不肯来参加舞会了。所以,请你编程计算,邀请哪些职员可以使快乐指数最大,求最大的快乐指数。
输入输出格式
输入格式:
第一行一个整数N。(1<=N<=6000)
接下来N行,第i+1行表示i号职员的快乐指数Ri。(-128<=Ri<=127)
接下来N-1行,每行输入一对整数L,K。表示K是L的直接上司。
最后一行输入0 0
输出格式:
输出最大的快乐指数。
输入输出样例
树形DP——DFS版
#include<bits/stdc++.h> #define N 6005
using namespace std; int n,r[N],dp[N][],tot,head[N];
struct node{
int to,next;
}e[N]; bool v[N];
void add(int u,int v){
e[++tot].to=v,e[tot].next=head[u],head[u]=tot;
}
//dp[i][0]表示i点不被选择时最大值
//dp[i][1]表示i点被选择时的最大值
void tredp(int u){
dp[u][]=,dp[u][]=r[u];
for(int i=head[u];i;i=e[i].next){
int v=e[i].to;
tredp(v);
dp[u][]+=max(dp[v][],dp[v][]);
dp[u][]+=dp[v][];
}
} int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&r[i]);
for(int a,b,i=;i<n;i++){
scanf("%d%d",&a,&b);
add(b,a);v[a]=;
}
int root;
for(int i=;i<=n;i++) if(!v[i]) root=i;
tredp(root);
printf("%d\n",max(dp[root][],dp[root][]));
return ;
}
树形DP——倒序队列或栈
#include<bits/stdc++.h> #define N 6005
using namespace std; int n,r[N],dp[N][],tot,head[N];
struct node{
int to,next;
}e[N]; bool v[N];
void add(int u,int v){
e[++tot].to=v,e[tot].next=head[u],head[u]=tot;
}
//dp[i][0]表示i点不被选择时最大值
//dp[i][1]表示i点被选择时的最大值
queue<int>Q;
stack<int>q;
bool vis[N];
void bfs(int root){
Q.push(root);
while(!Q.empty()){
int u=Q.front();Q.pop();q.push(u);
for(int i=head[u];i;i=e[i].next){
int v=e[i].to;
if(!vis[v]){
vis[v]=;
Q.push(v);
}
}
}
while(!q.empty()){
int u=q.top();q.pop();
for(int i=head[u];i;i=e[i].next){
int v=e[i].to;
dp[u][]+=max(dp[v][],dp[v][]);
dp[u][]+=dp[v][];
}dp[u][]+=r[u];
}
} int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&r[i]);
for(int a,b,i=;i<n;i++){
scanf("%d%d",&a,&b);
add(b,a);v[a]=;
}
int root;
for(int i=;i<=n;i++) if(!v[i]) root=i;
bfs(root);
printf("%d\n",max(dp[root][],dp[root][]));
return ;
}
拓扑排序——反向建边
#include<bits/stdc++.h> #define N 6005
using namespace std; int n,r[N],dp[N][],tot,head[N],rd[N];
struct node{
int to,next;
}e[N]; void add(int u,int v){
e[++tot].to=v,e[tot].next=head[u],head[u]=tot;
}
//dp[i][0]表示i点不被选择时最大值
//dp[i][1]表示i点被选择时的最大值
bool v[N];
queue<int>Q;
void topo(){
for(int i=;i<=n;i++) if(!rd[i]) Q.push(i);
while(!Q.empty()){
int u=Q.front();Q.pop();dp[u][]+=r[u];
for(int i=head[u];i;i=e[i].next){
int V=e[i].to;
rd[V]--;
if(!rd[V]) Q.push(V);
dp[V][]+=max(dp[u][],dp[u][]);
dp[V][]+=dp[u][];
}
}
}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&r[i]);
for(int a,b,i=;i<n;i++){
scanf("%d%d",&a,&b);
add(a,b);rd[b]++;v[a]=;
}
int root;
for(int i=;i<=n;i++) if(!v[i]) root=i;
topo();
printf("%d\n",max(dp[root][],dp[root][]));
return ;
}
P1352 没有上司的舞会——树形DP入门的更多相关文章
- P1352 没有上司的舞会&&树形DP入门
https://www.luogu.com.cn/problem/P1352 题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的 ...
- 洛谷P1352 没有上司的舞会——树形DP
第一次自己写树形DP的题,发个博客纪念`- 题目来源:P1352 没有上司的舞会 题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结 ...
- P1352 没有上司的舞会[树形dp]
题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司.现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数Ri, ...
- [luogu]P1352 没有上司的舞会[树形DP]
本Lowbee第一次写树形DP啊,弱...一个变量写错半天没看出来...... 题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点 ...
- 洛谷 P1352 没有上司的舞会 树形DP板子
luogu传送门 题目描述: 某大学有n个职员,编号为1~n. 他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司. 现在有个周年庆宴会,宴会每邀请来一个职员都会 ...
- 洛谷 P1352 没有上司的舞会(树形 DP)
题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司.现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数Ri, ...
- 『没有上司的舞会 树形DP』
树形DP入门 有些时候,我们需要在树形结构上进行动态规划来求解最优解. 例如,给定一颗\(N\)个节点的树(通常是无根树,即有\(N-1\)条无向边),我们可以选择任意节点作为根节点从而定义出每一颗子 ...
- CodeVS1380 没有上司的舞会 [树形DP]
题目传送门 没有上司的舞会 题目描述 Description Ural大学有N个职员,编号为1~N.他们有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司.每个职员有一个 ...
- 没有上司的舞会 树形dp
题目描述 某大学有N个职员,编号为1~N.他们之间有从属关系,也就是说他们的关系就像一棵以校长为根的树,父结点就是子结点的直接上司.现在有个周年庆宴会,宴会每邀请来一个职员都会增加一定的快乐指数Ri, ...
随机推荐
- [LeetCode][Java] Trapping Rain Water
题意: Given n non-negative integers representing an elevation map where the width of each bar is 1, co ...
- 5313 [JL]判断邮箱地址 升级版
5313 [JL]判断邮箱地址 升级版 时间限制: 1 s 空间限制: 1000 KB 题目等级 : 黄金 Gold 题解 查看运行结果 题目描述 Description 正确的邮箱地 ...
- Batch 拷贝远程机器文件到本机指定目录下
net use * /del /yesNET USE Y: \\远程机IP\d$ 登录密码 /user:domain\登录用户 set sourcePath="Y:\DOAutomatio ...
- 使用psutil模块获取电脑运行信息
psutil是python的一个用于获取cpu信息的模块,非常好使,以下附上官方的一些example: CPU-> Examples >>> import psutil > ...
- Linux基本命令 文件管理 上部
第1章 Linux入门相关 目录基本知识 Linux一切从根开始 倒挂的树形结构 对路径与相对路径 绝对路径: 从根开始的路径 比如:/oldboy /data 相对路径: 没有从根开始的路径 比如 ...
- 慕课网3-13编程练习:采用flex弹性布局制作页面主导航
小伙伴们,伸缩容器的属性我们已经学完了,接下来使用我们所学的伸缩容器属性完成下面的效果图. 要求: 1.logo.导航项.登录注册按钮这三项在水平和垂直方向上都对齐,而且他们之间的距离也相等. 2.导 ...
- web安全:防止浏览器记住或自动填写用户名和密码(表单)的终极解决方案
最近项目上要求做到这一点,在网上搜了一圈,发现都是不完美的,不兼容全部的浏览器,于是只能自己摸索了,最终得出了终极解决方案: 涉及: disabled 或 readonly display:none; ...
- javascript 获取时间
Js获取当前日期时间及其它操作 var myDate = new Date();myDate.getYear(); //获取当前年份(2位)myDate.getFullYear(); ...
- JavaScript学习四
2019-06-01 09:09:23 坚持,加油!!! 函数的学习 <html> <head> <script type="text/javascript&q ...
- vue学习记录(一)—— vue开发调试神器vue-devtools安装
网上有些贴子少了至关重要的一步导致我一直没装上, 切记!!install后还需build,且install和build都在vue-devtools文件夹内执行 github下载地址 点击跳转 具体步骤 ...