### 题面
https://www.lydsy.com/JudgeOnline/problem.php?id=4987
### 分析

先考虑贪心,求出树的直径,显然直径应该只被经过1次(最长的边应该走最少次数),其他非直径上的边被经过2次,整体的形状应该类似一条链上接着许多子树
考虑树形DP
子状态:$dp[x][i][j]( j\in \left\{0,1,2\right\})$,表示以x为根的子树中选了i个点,i个点中有j个直径的端点时的长度之和
其中x为树上一点,x为y的父亲,j为x子树去掉y子树后子树内选的点的个数,k为y子树内选的点的个数.接下来开始毒瘤的分类讨论
1. x,y的子树均不包含直径,边(x,y)属于子树内部的边,被算了2次$$dp[x][j+k][0]=\min(dp[x][j+k][0],dp[x][j][0]+dp[y][k][0]+2\cdot len);$$
2. y的子树包含直径的1个端点,则直径一定经过y,边(x,y)在直径上只被算1次$$dp[x][j+k][1]=\min(dp[x][j+k][1],dp[x][j][0]+dp[y][k][1]+len);$$
3. x的子树包含直径的1个端点,y的子树不包含直径的端点,通过边(x,y)与直径相连,被算2次$$dp[x][j+k][1]=\min(dp[x][j+k][1],dp[x][j][1]+dp[y][k][0]+2\cdot len);$$
4. x的子树包含直径的2个端点,y的子树不包含直径的端点,通过边(x,y)与直径相连,被算2次$$dp[x][j+k][2]=\min(dp[x][j+k][2],dp[x][j][2]+dp[y][k][0]+2\cdot len);$$
5. y的子树包含直径的2个端点,则直径一定不经过边(x,y),边(x,y)被算2次$$dp[x][j+k][2]=\min(dp[x][j+k][2],dp[x][j][0]+dp[y][k][2]+2\cdot len);$$
6. x,y的子树各包含直径的1个端点,直径一定过(x,y),边(x,y)被算1次$$dp[x][j+k][2]=\min(dp[x][j+k][2],dp[x][j][1]+dp[y][k][1]+len);$$

### 代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 3005
#define INF 0x3f3f3f3f3f3f3f3f
using namespace std;
inline int qread(){
int x=0,sign=1;
char c=getchar();
while(c<'0'||c>'9'){
if(c=='-') sign=-1;
c=getchar();
}
while(c>='0'&&c<='9'){
x=x*10+c-'0';
c=getchar();
}
return x*sign;
} int n,m;
struct edge {
int from;
int to;
int next;
long long len;
edge() { }
edge(int u,int v,long long w,int hd) {
from=u;
to=v;
next=hd;
len=w;
}
} E[maxn<<1];
int head[maxn];
int ecnt=0;
void add_edge(int u,int v,long long w) {
E[++ecnt]=edge(u,v,w,head[u]);
head[u]=ecnt;
} int sz[maxn];
long long dp[maxn][maxn][3];
void dfs(int x,int fa) {
sz[x]=1;
dp[x][1][0]=dp[x][1][1]=0;
for(int i=head[x]; i; i=E[i].next) {
int y=E[i].to;
long long len=E[i].len;
if(y!=fa) {
dfs(y,x);
//j类似01背包中的第二层,为防止被算多次,倒序循环
for(int j=min(sz[x],m); j>=0; j--) {//j为x子树去掉y子树后子树内选的点的个数
for(int k=min(sz[y],m-j); k>=0; k--) {//k为y子树内选的点的个数
dp[x][j+k][0]=min(dp[x][j+k][0],dp[x][j][0]+dp[y][k][0]+len*2);
//x,y的子树均不包含直径,边(x,y)属于子树内部的边,被算了2次
dp[x][j+k][1]=min(dp[x][j+k][1],dp[x][j][0]+dp[y][k][1]+len);
//y的子树包含直径的1个端点,则直径一定经过y,边(x,y)在直径上只被算1次
dp[x][j+k][1]=min(dp[x][j+k][1],dp[x][j][1]+dp[y][k][0]+len*2);
//x的子树包含直径的1个端点,y的子树不包含直径的端点,通过边(x,y)与直径相连,被算2次
dp[x][j+k][2]=min(dp[x][j+k][2],dp[x][j][2]+dp[y][k][0]+len*2);
//x的子树包含直径的2个端点,y的子树不包含直径的端点,通过边(x,y)与直径相连,被算2次
dp[x][j+k][2]=min(dp[x][j+k][2],dp[x][j][0]+dp[y][k][2]+len*2);
//y的子树包含直径的2个端点,则直径一定不经过边(x,y),边(x,y)被算2次
dp[x][j+k][2]=min(dp[x][j+k][2],dp[x][j][1]+dp[y][k][1]+len);
//x,y的子树各包含直径的1个端点,直径一定过(x,y),边(x,y)被算1次
}
}
sz[x]+=sz[y];
}
}
}
int main() {
// freopen("tree9.in","r",stdin);
int u,v;
long long w;
n=qread();
m=qread();
for(int i=1;i<n;i++){
u=qread();
v=qread();
w=qread();
add_edge(u,v,w);
add_edge(v,u,w);
}
memset(dp,0x3f,sizeof(dp));
dfs(1,0);
long long ans=INF;
for(int i=1;i<=n;i++){
ans=min(ans,dp[i][m][2]);//根据之前贪心的分析,直径一定经过这k个点中的一些点
}
cout<<ans;
}

BZOJ 4987 (树形DP)的更多相关文章

  1. BZOJ 1040 树形DP+环套树

    就是有n个点n条边,那么有且只有一个环那么用Dfs把在环上的两个点找到.然后拆开,从这条个点分别作树形Dp即可. #include <cstdio> #include <cstrin ...

  2. BZOJ - 2500 树形DP乱搞

    题意:给出一棵树,两个给给的人在第\(i\)天会从节点\(i\)沿着最长路径走,求最长的连续天数\([L,R]\)使得\([L,R]\)为起点的最长路径极差不超过m 求\(1\)到\(n\)的最长路经 ...

  3. BZOJ 4033 树形DP

    http://blog.csdn.net/mirrorgray/article/details/51123741 安利队长blog- 树形dp吧,状态挺显然的,dp[x][j]表示以x为根的子树中,选 ...

  4. [USACO10MAR]伟大的奶牛聚集 BZOJ 1827 树形dp+dfs

    题目描述 Bessie is planning the annual Great Cow Gathering for cows all across the country and, of cours ...

  5. bzoj 4007 树形dp

    题目大意 脸哥最近来到了一个神奇的王国,王国里的公民每个公民有两个下属或者没有下属,这种关系刚好组成一个 n 层的完全二叉树.公民 i 的下属是 2 * i 和 2 * i +1.最下层的公民即叶子节 ...

  6. BZOJ 1369 树形DP

    思路: f[i][j] 表示节点i 染成j时 子树的最小权值 (我会猜这个j很小 你打我吖~) 随便DP一发就好了 (证明我也不会) //By SiriusRen #include <cstdi ...

  7. [BZOJ 4033] [HAOI2015] T1 【树形DP】

    题目链接:BZOJ - 4033 题目分析 使用树形DP,用 f[i][j] 表示在以 i 为根的子树,有 j 个黑点的最大权值. 这个权值指的是,这个子树内部的点对间距离的贡献,以及 i 和 Fat ...

  8. [BZOJ 1907] 树的路径覆盖 【树形DP】

    题目链接:BZOJ - 1907 题目分析 使用树形 DP,f[x][0] 表示以 x 为根的子树不能与 x 的父亲连接的最小路径数(即 x 是一个折线的拐点). f[x][1] 表示以 x 为根的子 ...

  9. bzoj 4871: [Shoi2017]摧毁“树状图” [树形DP]

    4871: [Shoi2017]摧毁"树状图" 题意:一颗无向树,选两条边不重复的路径,删去选择的点和路径剩下一些cc,求最多cc数. update 5.1 : 刚刚发现bzoj上 ...

随机推荐

  1. HTML与CSS中的颜色与单位个人分享

    颜色与单位 Web安全色有216中其中色彩有210中,非色彩6中 前景色与背景色 前景色就是设置字体的颜色 背景色就是为指定元素设置背景色 - 浏览器默认背景色的颜色为透明色 颜色的命名 1.使用单词 ...

  2. Python---常用的内置模块

    #fsum() 对整个序列求和   返回浮点数 print(math.fsum([1,4.5,5,7])) #sum() python内置求和 print(sum([1,4,5,7])) print( ...

  3. 【GDOI 2016 Day1】疯狂动物城

    题目 分析 注意注意:码农题一道,打之前做好心理准备. 对于操作1.2,修改或查询x到y的路径,显然树链剖分. 对于操作2,我们将x到y的路径分为x到lca(x,y)和lca(x,y)到y两部分. 对 ...

  4. vue 常见错的可能原因

    标签或者组件名写错 Unknown custom element: <h> - did you register the component correctly? For recursiv ...

  5. IDEA提交代码到github

    GIT客户端安装及idea配置github账号并提交代码到GIT参考资料:https://blog.csdn.net/qq_31405633/article/details/88193119 1. 选 ...

  6. 3D打印格式STL

    STL格式及其转换 近期接触了3D打印的一些东西,也制作了一个vrml转stl的插件,对该领域多了一些认识. 目前尚没有打印机直接支持stl.obj等格式,在打印之前需使用厂家提供的软件将stl等格式 ...

  7. Codeforces 980D

    这题其实挺水的,但我比较vegetable,交了好多次才过. 题意: 给定一个序列,把这个序列的所有连续子序列分组,每组中任意两个数相乘是个完全平方数,输出每个子序列最少分的组数: 思路: 先把每个数 ...

  8. nginx之Geoip读取地域信息模块

    1 geoip_module模块 基于IP地址匹配MaxMind GeolP二进制文件,读取IP所在地域信息. yum install nginx-module-geoip geoip2已经有了,安装 ...

  9. JPA Hibernate 等面向对象持久化框架难以解决的问题

    1+N查询性能问题 单个更新实体对象,而不是使用单调语句 在Java端做了很重的数据处理,数据库端反而很少 多层次 fetch inner join 难以实现 关联映射难以自定义条件 级联操作难以掌控 ...

  10. python-之基本语法

    模块一些函数和类的集合文件,并实现一定的功能,当我们需要使用这些功能的时候,可以直接把相应的模块导入到我们的程序中 import import mode    #导入mode模块 即导入mode模块后 ...