[vijos1144]小胖守皇宫<树形dp>
题目链接:https://vijos.org/p/1144
woc我竟然A了,这道经典的树形dp或者说是树形dp的入门题我终于过了,虽然之前做过一些树形dp的题,但是这题开始还是一脸懵逼,dp方程如何定义都知道,但是不懂转移啊,这就有点伤了。。
dp方程定义dp[i][1]节点i 选自己
dp[i][2]节点i选自己的儿子==不选自己和父亲
dp[i][3]节点i选自己的父亲==不选自己选父亲
然后就是转移了。。毕竟是基础题嘛,所以转移也不难
转移的时候我们是直接递归到叶节点然后再做前面的。。所以我们不用考虑父节点的状态
dp[i][1]选自己的时候,儿子节点就有两种方式,选儿子自己,或者选儿子的父亲dp[i][1]+=min(dp[son][1],dp[son][3]);
dp[i][2]选儿子时 ,儿子就选或不选两个方式,但是如果一旦所有的儿子都是不选了,我们就要找一个最小的儿子树的值在最后加上,
dp[i][2]+=min(dp[son][1],dp[son][2])如果全部选了2,就要在结尾加上dp[i][2]+=min(dp[son][1]-dp[son][2])
至于为啥加这个,就是这道题唯一有点思考难度的地方了,因为你是要加最小的儿子选一个的值,所以找到最小,比如时第s2个儿子,之前已经加了dp[s2][2],所以最后的时候是加上dp[s2][1]-dp[s2][2],相当于把之前的那个dp[s2][2]抵消了,就不会加重复
dp[i][3]选父亲时,因为我们是从子往父推,所以不从父亲转移,就考虑这时候的儿子节点,儿子节点来源就是儿子自己放和儿子的儿子放
dp[i][3]+=min(dp[son][1],dp[son][2]);
好吧这就是这道题的全部了,最后只需要输出根节点的选自己和选儿子方案的最小值就可,因为根没有父亲。。。
储存这个关系的方式有两种,一种是链表,一种是多叉树转二叉树,
两种方式的不同点在于链表是双向的,然后重新建树,默认1为根节点
而多叉树转二叉树是以题目给的关系建树,以没有父亲的点为根
链表:
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cstdlib>
#include<cmath>
#include<queue>
#define maxn 1505
using namespace std; int f[maxn][],n;
struct edge{
int u,v,w,nxt;
}e[maxn*];
int a[maxn],head[maxn],vis[maxn],tot; void adde(int u,int v){
tot++;
e[tot].u=u;e[tot].v=v;
e[tot].nxt=head[u];
head[u]=tot;
} void work(int x){
f[x][]=a[x];
int s=0x3f3f3f,p=;
for(int i=head[x];i!=-;i=e[i].nxt){
int v=e[i].v;
if(vis[v]==)continue;
vis[v]=;
work(v);
f[x][]+=min(f[v][],f[v][]);
if(f[v][]<f[v][]){
f[x][]+=f[v][],p=;
}else{
f[x][]+=f[v][],s=min(s,f[v][]-f[v][]);
}
f[x][]+=min(f[v][],f[v][]);
}
if(p==)f[x][]+=s;
} int main(){
memset(head,-,sizeof(head));
scanf("%d",&n);
for(int i=;i<=n;i++){
int num,val,sum;
scanf("%d%d%d",&num,&val,&sum);
a[num]=val;
for(int j=;j<=sum;j++){
int b;
scanf("%d",&b);
adde(num,b);adde(b,num);
}
}
vis[]=;work();
printf("%d",min(f[][],f[][]));
}
多叉树转二叉树
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cstdlib>
#include<cmath>
#include<queue>
#define maxn 3005
using namespace std; int n,m,a[maxn],root;
int f[maxn][],vis[maxn];
int lson[maxn],rson[maxn],fa[maxn]; void work(int x)
{
f[x][]=a[x];
int s=0x3f3f3f,p=;
for(int i=lson[x];i!=;i=rson[i]){
work(i);
if(f[i][]<f[i][]){
f[x][]+=f[i][];s=min(f[i][]-f[i][],s);
}else f[x][]+=f[i][],p=;
f[x][]+=min(f[i][],f[i][]);
f[x][]+=min(f[i][],f[i][]);
}
if(p==)f[x][]+=s;
} int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++){
int num,val,q;
scanf("%d%d%d",&num,&val,&q);
a[num]=val;
for(int j=;j<=q;j++){
int s,now=lson[num];
scanf("%d",&s);
fa[s]=num;
if(j==)lson[num]=s;
else {
while(rson[now]!=){
now=rson[now];
}
rson[now]=s;
}
}
}
for(int i=;i<=n;i++)
if(fa[i]==)work(i),root=i;
printf("%d",min(f[root][],f[root][]));
}
提醒一点,多叉树转二叉树需要在执行dp之前跑个O(n)找到根节点
[vijos1144]小胖守皇宫<树形dp>的更多相关文章
- 【树形dp】vijos1144小胖守皇宫
细节很精妙 描述 huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫. 皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:某些宫殿间可以互相望见.大内保卫森严,三步一岗,五步 ...
- Vijos1144小胖守皇宫【树形DP】
皇宫看守 太平王世子事件后,陆小凤成了皇上特聘的御前一品侍卫.皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:某些宫殿间可以互相望见.大内保卫森严,三步一岗,五步一哨,每个宫殿都要有人全天候看 ...
- Vijos 1144 小胖守皇宫 【树形DP】
小胖守皇宫 描述 huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫. 皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:某些宫殿间可以互相望见.大内保卫森严,三步一岗,五步 ...
- 树形dp 之 小胖守皇宫
题目描述 huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫. 皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:有边相连的宫殿间可以互相望见.大内保卫森严,三步一岗,五步一 ...
- 小胖守皇宫(VIJOS P1144 )题解
题目描述 huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫. 皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:某些宫殿间可以互相望见.大内保卫森严,三步一岗,五步一哨,每 ...
- vijos 小胖守皇宫
点击打开题目 树形DP 显然会想到某个点放或不放守卫来定义状态,但在不放的情况下,需要分类讨论是父亲放还是一个儿子放,于是定义以下状态: f[root][0]表示自己不放,父亲也不放 f[root][ ...
- 【vijos1144】小胖守皇宫(树形DP)
描述 huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫. 皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:某些宫殿间可以互相望见.大内保卫森严,三步一岗,五步一哨,每个宫 ...
- vijos1144(小胖守皇宫)
也是ural1039 描述 huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫. 皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状:某些宫殿间可以互相望见.大内保卫森严,三步 ...
- 树形DP 复习
树形DP 树形DP:建立在树上的动态规划 一般有两种传递方式:根→叶或叶→根 前者出现在换根DP中,一般操作是求出某一个点的最优解,再通过这一个点推知其他点的最优解. 后者是树形DP的常见形式,一般树 ...
随机推荐
- 简单易懂的Servlet路径问题
关于servlet路径,我看了一下网上别人的博客园,发现都有一个通病,讲的太专业了,又抓不住关键部分,往往看一眼就不想看第二眼.所以我特地准备了初学者所通识的servlet路径问题. 1.标识符 /j ...
- ASP.NET Core 快速入门(Razor Pages + Entity Framework Core)
引子 自从 2009 年开始在博客园写文章,这是目前我写的最长的一篇文章了. 前前后后,我总共花了 5 天的时间,每天超过 3 小时不间断写作和代码调试.总共有 8 篇文章,每篇 5~6 个小结,总截 ...
- Lucene查询语法汇总
目录 一.单词查询 二.通配符查询 三.模糊查询 四.近似查询 五.范围查询 六.优先级查询 七.逻辑操作 八.括号分组 九.转义特殊字符 Lucene是目前最为流行的开源全文搜索引擎工具包,提供了完 ...
- 原生js实现随着滚动条滚动,导航会自动切换的效果
最近学习前端,把前面用原生js写的一段有关tab切换效果的代码贴上,实现的效果比较简陋,请大家见谅 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1 ...
- Simulink仿真入门到精通(十八) TLC语言
TLC(Target Language Compiler)是一种为转换为目标语言而存在的额解释性语言,其目的就是将模型中编译出来的rtw文件转换为目标代码(C/C++等).与M语言类似,既可以写成脚本 ...
- centeos安装Anconda3
步骤: #获取安装包 wget https://mirrors.tuna.tsinghua.edu.cn/anaconda/archive/Anaconda3-5.2.0-Linux-x86_64.s ...
- div中单行文字垂直水平居中
1.div中单行文字垂直水平居中.条件:外层div高度已经给定.代码如下: 复制代码代码如下: <style type="text/css">.div3{border: ...
- XiaoQi.Study 项目(三)
一.配置跨域 1.首先注册跨域要求 ,(可访问的IP.端口) //注册跨域 services.AddCors(options => { options.AddPolicy("XiaoQ ...
- 何为内存模型(JMM)?
前言 任何一门语言都有其语言规范,从逻辑上我们可划分为语法规范和语义规范,语法规范则是描述了如何通过相关语法编写可执行的程序,而语义规范则是指通过语法编写的程序所构造出的具体含义.语言只要具备存储(比 ...
- JavaScript 原型与继承
JavaScript 原型与继承 JavaScript 中函数原型是实现继承的基础.prototype.construct.原型链以及基于原型链的继承是面向对象的重要内容 prototype 原型即 ...