洛谷P4362 贪吃的九头龙
大意就是把一棵树的点染成m种颜色,其中1号点的颜色必须染恰好k个节点。
总代价是所有两端点颜色相同的边的边权。
求最小代价。
解:可以分为m == 2和m > 2两个题。
m > 2时有代价的边的两端点显然是一号点色的(设为白色)。
m == 2的时候还要计算两端点是另外一种颜色的边的贡献(黑色)。
状态设计就是f[x][j][0/1]表示x为根的子树中染了j个白色点,x号点染/不染的最小代价。
转移的时候做一个类似树上背包的转移即可。
注意m == 2的时候,更新f[i][j][0]合并子树的时候要把原来的那个值覆盖掉,因为子节点也是0的时候会有代价,所以不能保留原来的没有计算这个代价的值。
我比较菜,一开始没发现要分成两个题,就写了两个DFS函数...
#include <cstdio>
#include <algorithm>
#include <cstring> const int N = ; struct Edge {
int nex, v, len;
}edge[N << ]; int top; int f[N][N][], e[N], n, k; inline void add(int x, int y, int z) {
top++;
edge[top].v = y;
edge[top].len = z;
edge[top].nex = e[x];
e[x] = top;
return;
} void DFS_2(int x, int fa) {
f[x][][] = ;
f[x][][] = ;
for(int i = e[x]; i; i = edge[i].nex) {
int y = edge[i].v;
if(y == fa) {
continue;
}
DFS_2(y, x);
for(int j = k; j >= ; j--) {
/// f[x][j] [0/1]
int t = 0x3f3f3f3f;
for(int p = j; p >= ; p--) {
t = std::min(t, std::min(f[y][p][] + f[x][j - p][], f[y][p][] + f[x][j - p][] + edge[i].len));
}
f[x][j][] = t;
t = 0x3f3f3f3f;
for(int p = j; p >= ; p--) {
t = std::min(t, std::min(f[y][p][] + f[x][j - p][] + edge[i].len, f[y][p][] + f[x][j - p][]));
}
f[x][j][] = t;
/*for(int p = j; p >= 0; p--) {
f[x][j][0] = std::min(f[x][j][0], f[y][p][0] + f[x][j - p][0] + edge[i].len);
f[x][j][0] = std::min(f[x][j][0], f[y][p][1] + f[x][j - p][0]);
if(j != p) {
if(x == 1 && j == 2)printf("step 0 f[1][2][1] = %d \n", f[1][2][1]);
f[x][j][1] = std::min(f[x][j][1], f[y][p][0] + f[x][j - p][1]);
if(x == 1 && j == 2)printf("step 1 f[1][2][1] = %d \n", f[1][2][1]);
f[x][j][1] = std::min(f[x][j][1], f[y][p][1] + f[x][j - p][1] + edge[i].len);
if(x == 1 && j == 2)printf("step 2 f[1][2][1] = %d \n", f[1][2][1]);
if(x == 1 && j == 4 && y == 2 && p == 2) {
printf("%d + %d \n", f[y][p][0] + f[x][j - p][1]);
}
if(x == 1 && j == 2) {
printf("> f 1 2 1 = %d p = %d \n", f[1][2][1], p);
printf("> %d + %d \n", f[y][p][0], f[x][j - p][1]);
printf("> %d + %d \n", f[y][p][1], f[x][j - p][1] + edge[i].len);
}
}
}*/
}
}
/*for(int j = 0; j <= k; j++) {
printf("f %d %d %d = %d \n", x, j, 0, f[x][j][0]);
*/
return;
} void DFS_1(int x, int fa) {
f[x][][] = ;
f[x][][] = ;
for(int i = e[x]; i; i = edge[i].nex) {
int y = edge[i].v;
if(y == fa) {
continue;
}
DFS_1(y, x);
//
for(int j = k; j >= ; j--) {
/// f[x][j] [0/1]
for(int p = j; p >= ; p--) {
f[x][j][] = std::min(f[x][j][], f[y][p][] + f[x][j - p][]);
f[x][j][] = std::min(f[x][j][], f[y][p][] + f[x][j - p][]);
if(p != j) {
f[x][j][] = std::min(f[x][j][], f[y][p][] + f[x][j - p][]);
f[x][j][] = std::min(f[x][j][], f[y][p][] + f[x][j - p][] + edge[i].len);
}
}
}
}
return;
} int main() {
int m;
memset(f, 0x3f, sizeof(f));
scanf("%d%d%d", &n, &m, &k);
for(int i = , x, y, z; i < n; i++) {
scanf("%d%d%d", &x, &y, &z);
add(x, y, z);
add(y, x, z);
}
if(n - k < m - ) {
puts("-1");
return ;
}
if(m > ) {
DFS_1(, );
printf("%d\n", f[][k][]);
}
else {
DFS_2(, );
printf("%d\n", f[][k][]);
}
return ;
}
AC代码
洛谷P4362 贪吃的九头龙的更多相关文章
- Vijos1523贪吃的九头龙【树形DP】
贪吃的九头龙 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头 ...
- [codevs1746][NOI2002]贪吃的九头龙
[codevs1746][NOI2002]贪吃的九头龙 试题描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时 ...
- 贪吃的九头龙(tyvj P1523)
T2 .tyvj P1523贪吃的九头龙 描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于 ...
- Vijos 1523 贪吃的九头龙 【树形DP】
贪吃的九头龙 背景 安徽省芜湖市第二十七中学测试题 NOI 2002 贪吃的九头龙(dragon) Description:OfficialData:OfficialProgram:Converted ...
- [NOI2002]贪吃的九头龙(树形dp)
[NOI2002]贪吃的九头龙 题目背景 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是 说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的 ...
- codevs1746 贪吃的九头龙
[问题描述]传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落.有一 ...
- vojis1523 NOI2002 贪吃的九头龙
描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落. 有一天, ...
- codevs贪吃的九头龙
传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落.有一天,有M 个 ...
- [NOI2002] 贪吃的九头龙
题目类型:树形DP 传送门:>Here< 题意:有一只九头龙要吃了一颗树,给出一棵\(N\)个节点的带边权的树.九头龙有\(M\)个头,其中一个是大头,大头要吃恰好\(K\)个节点,其他头 ...
随机推荐
- Vue实例:演示input 和 textarea 元素中使用 v-model 实现双向数据绑定
最终效果: 主要代码: <template> <div> <p>input 元素:</p> <input v-model="messag ...
- vue中的跨域问题
https://segmentfault.com/a/1190000011072725(原文) 使用vue-axios和vue-resource解决vue中调用网易云接口跨域的问题 注(api很重 ...
- mac下php开发环境的搭建
1.phpstorm 在官网:https://www.jetbrains.com/phpstorm/,下载最新版:phpstorm-2016.2.1 在http://15.idea.lanyus.co ...
- 为什么js中要用void 0 代替undefined
这个是Backbone.js中的一句源码 if (callback !== void 0 && 'context' in opts && opts.context == ...
- CSS实现元素水平垂直居中
我们知道,实现元素的水平居中比较简单,在设置了宽度后,设置左右margin为auto就可以. 但是如何设置元素垂直居中呢? 当然,对于单行的文字,可以通过设置line-height来解决, 可以对于一 ...
- 版本控制--git+idea
- 为linux主机增加file description
在benchmarked写的服务器的时候就遇到了too many file open 这个报错. 由于遇到过很多次了,所以知道应该是单机fd打满了. 首先来看看 机器最多支持多少fd cat /pro ...
- pip 安装 MySQL-python 报错
报错一:EnvironmentError: mysql_config not found 解决:yum install mysql-devel 报错二:Python.h No such file or ...
- 转 Pycharm及python安装详细教程
转 : http://blog.csdn.net/qq_29883591/article/details/52664478 首先我们来安装Python 1.首先进入网站下载:点击打开链接(或自己输入网 ...
- 【数学建模】day07-数理统计II
方差分析和回归分析. 用数理统计分析试验结果.鉴别各因素对结果影响程度的方法称为方差分析(Analysis Of Variance),记作 ANOVA. 比如:从用不同工艺制作成的灯泡中,各自抽取了若 ...