[NOI2002] 贪吃的九头龙
题目类型:树形DP
传送门:>Here<
题意:有一只九头龙要吃了一颗树,给出一棵\(N\)个节点的带边权的树。九头龙有\(M\)个头,其中一个是大头,大头要吃恰好\(K\)个节点,其他头吃几个随意。如果一个头吃了一个连通块,那么他们会把树枝也吃下去,获得边权那么多的难受值。先要吃完整棵树,使难受值总和最小
解题思路
首先会发现题目蕴含着一个奇妙的性质。会得到一条边的难受值当且仅当一条边相邻的两个节点是被同一个头吃的,换句话说如果相邻的两个节点是不同的龙吃的那么就不会获得该边的难受值。当我们确定大头吃的\(K\)个点之后,由于剩下的头每个头想吃几个就吃几个,由于给出了一颗树,一定存在一种方案使得任意两个相邻的点是不同的头。事实上,只需要多余的两种头就可以了。换句话说,我们可以把所有多余\(3\)个头的情况看做是\(3\)个头的。
当然需要特殊考虑\(M=2\)的情况,此时剩余的只有一个头了,需要特殊处理
于是,难受值完全取决于大头的那\(K\)个节点如何选择。
考虑树形DP:\(f[u][j][0/1]\)表示子树\(u\)中大头吃\(j\)个的最小难受值之和,其中\(k=0\)表示大头不吃根节点,反之亦然
于是可以发现我们只需要判断\(u,v\)的颜色是否相同即可以转移:当\(M>2\)时,除非\(u,v\)都是\(1\),否则不考虑。当\(M=2\)时,\(0,1\)可以直接表示他们的头了,所以也就是\(u ⊕ v\)
于是我们可以初步得到转移方程:$$f[u][j][1]=Min{f[v][k][0]+g[u][j-k][1],f[v][k][1]+g[u][j-k][1]+cost(u,v)}$$$$f[u][j][0]=Min{f[v][k][0]+g[u][j-k][0]+cost(u,v)*[M=2],f[v][k][1]+g[u][j-k][0]}$$
注意方程中的\(g\)数组,我们好像并没有涉及到它。事实上我们发现,方程的转移时需要枚举\(k\)作为中介的,但是众多儿子,每个儿子的\(k\)如何分配?题目给出了一颗多叉树,使得这一步非常复杂。
联想二叉树的做法,二叉树时当确定一个儿子是\(k\)时,马上就能确定另一个儿子是\(j-k\)。如果也用这种方式来处理多叉树就好了。于是我们在考虑第\(j\)个儿子的时候,可以记录前\(j-1\)个已经做过的儿子的最优值。而上一轮留下的结果正是\(f\)数组!但由于\(f\)数组在不断的更新,肯定不能直接拿过来用,因此需要一个临时数组\(g\)来记录上一轮的结果,其实形象的理解,就是把前面的\(j-1\)个儿子并在了一起
由于每一轮\(f\)数组都要重新更新,因此每一轮都要重新赋值\(+∞\)。另外很容易得到另个初始化条件:\(f[u][0][0]=1, f[u][1][1]=0\)。意义都很显然
Code
/*By DennyQi 2018.8.20*/
#include <cstdio>
#include <queue>
#include <cstring>
#include <algorithm>
#define r read()
#define Max(a,b) (((a)>(b)) ? (a) : (b))
#define Min(a,b) (((a)<(b)) ? (a) : (b))
using namespace std;
typedef long long ll;
const int MAXN = 310;
const int MAXM = 610;
const int INF = 1061109567;
inline int read(){
int x = 0; int w = 1; register int c = getchar();
while(c ^ '-' && (c < '0' || c > '9')) c = getchar();
if(c == '-') w = -1, c = getchar();
while(c >= '0' && c <= '9') x = (x<<3) + (x<<1) + c - '0', c = getchar();return x * w;
}
int N,M,K,x,y,z;
int first[MAXM],nxt[MAXM],to[MAXM],cost[MAXM],cnt;
int f[MAXN][MAXN][2], g[MAXN][2];
inline void add(int u, int v, int w){
to[++cnt]=v, cost[cnt]=w, nxt[cnt]=first[u], first[u]=cnt;
}
void DP(int u, int _f){
int v;
f[u][0][0] = f[u][1][1] = 0;
for(int i = first[u]; i; i = nxt[i]){
if((v = to[i]) == _f) continue;
DP(v, u);
for(int j = 0; j <= K; ++j){
g[j][0] = f[u][j][0];
g[j][1] = f[u][j][1];
f[u][j][0] = INF;
f[u][j][1] = INF;
}
for(int j = 0; j <= K; ++j){
for(int k = 0; k <= j; ++k){
f[u][j][1] = Min(f[u][j][1], f[v][k][0]+g[j-k][1]);
f[u][j][1] = Min(f[u][j][1], f[v][k][1]+g[j-k][1]+cost[i]);
if(M == 2){
f[u][j][0] = Min(f[u][j][0], f[v][k][0]+g[j-k][0]+cost[i]);
}
else{
f[u][j][0] = Min(f[u][j][0], f[v][k][0]+g[j-k][0]);
}
f[u][j][0] = Min(f[u][j][0], f[v][k][1]+g[j-k][0]);
}
}
}
}
int main(){
N = r, M = r, K = r;
if(M-1 + K > N){
printf("-1");
return 0;
}
for(int i = 1; i < N; ++i){
x = r, y = r; z = r;
add(x, y, z);
add(y, x, z);
}
memset(f, 0x3f, sizeof(f));
DP(1, 0);
printf("%d", f[1][K][1]);
return 0;
}
[NOI2002] 贪吃的九头龙的更多相关文章
- [codevs1746][NOI2002]贪吃的九头龙
[codevs1746][NOI2002]贪吃的九头龙 试题描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时 ...
- [NOI2002]贪吃的九头龙(树形dp)
[NOI2002]贪吃的九头龙 题目背景 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是 说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的 ...
- vojis1523 NOI2002 贪吃的九头龙
描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落. 有一天, ...
- Vijos1523 NOI2002 贪吃的九头龙 树形dp
思路不算很难,但细节处理很麻烦 前面建图.多叉转二叉,以及确定dp处理序列的过程都是套路,dp的状态转移过程以注释的形式阐述 #include <cstdio> #include < ...
- 洛谷 P4362 [NOI2002]贪吃的九头龙
https://www.luogu.org/problemnew/show/P4362 首先有个很显然的dp:ans[i][j][k]表示i节点用j号头,i节点为根的子树中共有k个点用大头时i节点为根 ...
- Vijos1523贪吃的九头龙【树形DP】
贪吃的九头龙 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头 ...
- 贪吃的九头龙(tyvj P1523)
T2 .tyvj P1523贪吃的九头龙 描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于 ...
- Vijos 1523 贪吃的九头龙 【树形DP】
贪吃的九头龙 背景 安徽省芜湖市第二十七中学测试题 NOI 2002 贪吃的九头龙(dragon) Description:OfficialData:OfficialProgram:Converted ...
- codevs1746 贪吃的九头龙
[问题描述]传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落.有一 ...
随机推荐
- Jenkins-job之间依赖关系配置
使用场景: 想要在某APP打新包之后,立即执行自动化测试的job来验证该新包. 比如Job A 执行完执行Job B ,如下图所示,如何建立依赖呢? 1.配置上游依赖 构建触发器-配置如下信息: 选择 ...
- struts2之配置文件struts.xml详解
struts配置文件 struts.xml配置参数详解 struts.xml中很大一部分配置默认配置就好了 但是有些还是需要做了解 以便于理解 和修改 <?xml version=" ...
- 助教总结 -【福大软工实践-2017-2018-K班】
助教总结 -[福大软工实践-2017-2018-K班] 非常抱歉这么晚才来写总结! 助教工作 助教共发表博客39篇. 助教共点评约500条. 起步 对于常规课程的起步,通常都是在第一次课堂上由老师对课 ...
- OSS网页上传和断点续传(STSToken篇)
云账号AccessKey拥有所有API访问权限,在客户端不要直接使用,会泄露ak信息,造成安全问题.所以使用STS方式(临时账号权限)给客户端授权. C#版获取STSToken 一.下载阿里SDK(a ...
- 使用VBA进行JS加密的反混淆,还原JS代码。
本文地址:http://www.cnblogs.com/Charltsing/p/JSEval.html 联系QQ:564955427 类似下面的代码是登陆 全国企业信用信息公示系统(安徽)(网址:h ...
- (FZU 2150) Fire Game (bfs)
题目链接:http://acm.fzu.edu.cn/problem.php?pid=2150 Problem Description Fat brother and Maze are playing ...
- tortoisegit密钥与git密钥配置
在客户端生成密钥并将公钥上传到服务器可以避免每次连接git服务器都要登录的尴尬. 但git的私钥是不能直接用在tortoisegit上的,需要用tortoisegit的puttygen转换一下,详细过 ...
- #Leetcode# 942. DI String Match
https://leetcode.com/problems/di-string-match/ Given a string S that only contains "I" (in ...
- openstack,docker,mesos,k8s关系
openstack,docker,mesos,k8s什么关系? - 知乎https://www.zhihu.com/question/62985699 OpenStack + K8S 环境集成测试ht ...
- JS学习笔记 等于和包装对象
严格等于 a===b 首先判断两边数据的类型,若类型不同,返回false. 若类型相同(1.2和1.2,字符串相等指内容和长度都是一样的),返回true null===null undefined== ...