前言:寒假讲过树形DP,这次再复习一下。

--------------

基本的树形DP

实现形式

树形DP的主要实现形式是$dfs$。这是因为树的特殊结构决定的——只有确定了儿子,才能决定父亲。划分阶段的话一般是$f[i][j][0/1]$。$i$表示以$i$为根的子树,$j$一般表示保留$j$个子节点,$0/1$表示选/不选这个节点。一般第三维可以省去。

基本的DP方程

选择节点类

$f[i][0]=f[j][1]$

$f[i][1]=max/min(f[j][0],f[j][1])$

背包类

$f[v][k]=f[u][k]+val$

$f[u][k]=max(f[u][k],f[v][k-1])$

例题

没有上司的舞会

题目链接

设$f[i][0/1]$表示$i$这个人去/不去舞会能获得的最大快乐指数。

状态转移方程:

$f[u][0]=max(f[v][0],f[v][1])$

$f[u][1]=val[u]+f[v][0]$

最大子树和

题目链接

设$f[i]$表示以$i$为根的子树所能获得的最大美丽指数。

状态转移方程:$f[u]=val[u]+max(f[v],0)$

注意有负数。

树形DP求树的直径

树的直径,即树上最远点对,一般有两种做法:两次$dfs$和树形$DP$。两种方法这里都会讲一下。

两次$dfs$

步骤:我们先找任意一点的最远点$p$,再找$p$的最远点$w$。从$p$到$w$的距离就是树的直径。

证明:

①若$P$已经在直径上,根据树的直径的定义可知$Q$也在直径上且为直径的一个端点

②若$P$不在直径上,我们用反证法,假设此时$WQ$不是直径,$AB$是直径

--->若$AB$与$PQ$有交点$C$,由于$P$到$Q$最远,那么$PC+CQ>PC+CA$,所以$CQ>CA$,易得$CQ+CB>CA+CB$,即$CQ+CB>AB$,与$AB$是直径矛盾,不成立。

--->若$AB$与$PQ$没有交点,$M$为$AB$上任意一点,$N$为$PQ$上任意一点。首先还是$NP+NQ>NQ+MN+MB$,同时减掉$NQ$,得$NP>MN+MB$,易知$NP+MN>MB$,所以$NP+MN+MA>MB+MA$,即$NP+MN+MA>AB$,与$AB$是直径矛盾,所以这种情况也不成立。

代码:

//
#include<bits/stdc++.h>
using namespace std;
int dis[],ans,p;
struct node
{
int next,to,dis;
}edge[];
int head[],cnt;
inline void add(int from,int to,int dis)
{
edge[++cnt].next=head[from];
edge[cnt].to=to;
edge[cnt].dis=dis;
head[from]=cnt;
}
void dfs(int now,int fa)
{
if (ans<dis[now])
{
ans=dis[now];p=now;
}
for (int i=head[now];i;i=edge[i].next)
{
int to=edge[i].to;
if (to==fa) continue;
dis[to]=dis[now]+edge[i].dis;
dfs(to,now);
}
}
void find(int x)
{
ans=;
dis[x]=;
dfs(x,);
}
int main()
{
cin>>n>>m;
for (int i=;i<=m;i++)
{
int u,v,d;
cin>>u>>v>>d;
add(u,v,d);add(v,u,d);
}
find();
find(p);
cout<<ans;
return ;
}

树形DP

设$dis1[i]$和$dis2[i]$表示以$i$为根的子树中以$i$为起点的最大距离和次大距离。

状态转移方程:

$if (dis1[i]<dis1[j]+edge[i].dis) dis2[i]=dis1[i],dis1[i]=dis1[j]+edge[i].dis$

$else if (dis2[i]<dis1[j]+edge[i].dis) dis2[i]=dis1[j]+edge[i].dis$。

理解:这样做就是,先看能否更新最大值,若能,它的次大值就是原先的最大值,再更新它的最大值;若不能,就看能不能更新次大值,若能,就更新,不能就不管它。

$ans=\max {dis1[i]+dis2[i]}$

代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=;
int n,m,t,ans;
int f1[N],f2[N];
int first[N],v[N],w[N],next[N];
void add(int x,int y,int z){
t++;
next[t]=first[x];
first[x]=t;
v[t]=y;
w[t]=z;
}
void dp(int x,int father)
{
int i,j;
for(i=first[x];i;i=next[i])
{
j=v[i];
if(j==father) continue;
dp(j,x);
if(f1[x]<f1[j]+w[i])
{
f2[x]=f1[x];
f1[x]=f1[j]+w[i];
}
else if(f2[x]<f1[j]+w[i])
f2[x]=f1[j]+w[i];
ans=max(ans,f1[x]+f2[x]);
}
}
int main()
{
int x,y,z,i;
scanf("%d%d",&n,&m);
for(i=;i<=m;++i)
{
scanf("%d%d%d",&x,&y,&z);
add(x,y,z);
add(y,x,z);
}
dp(,);
printf("%d",ans);
return ;
}

树形DP求树的重心 

对于一棵$n$个节点的无根树,找到一个点,使得把树变成以该点为根的有根树时,最大子树的节点数最小。换句话说,删除这个点后最大连通块的节点数最小,那么这个点就是树的重心。

解法:任选一个节点为根,把无根树变成有根树,然后设$f[i]$表示以$i$为根的子树的节点个数。不难发现$f[i]=\sum \limits_{j\in s[i]}f[j]+1$。程序实现很简单:只需要一次$dfs$,在无根树转化成有根树的同时计算即可。其实在删除节点$i$后,最大的连通块有多少个节点呢?节点$i$的子树中最大的有$max{f[j]}$个节点,$i$的上方子树中有$n-f[i]$个节点,在$DP$过程中根据定义就可以找出树的重心了。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <algorithm>
#include <vector>
using namespace std; vector<int> G[];
int dp[],tot[];
vector<int> ans; void dfs(int u)
{
tot[u] = ;
for(int i=;i<G[u].size();i++)
{
int v = G[u][i];
if(tot[v] == -)
dfs(v);
else
continue;
dp[u] = max(dp[u],tot[v]);
tot[u] += tot[v];
}
} int main()
{
int n,i,j,u,v;
scanf("%d",&n);
for(i=;i<=n;i++)
G[i].clear();
for(i=;i<n-;i++)
{
scanf("%d%d",&u,&v);
G[u].push_back(v);
G[v].push_back(u);
}
memset(dp,,sizeof(dp));
memset(tot,-,sizeof(tot));
dfs();
int mini = Mod;
for(i=;i<=n;i++)
{
int tmp = max(dp[i],n-tot[i]);
if(tmp < mini)
{
ans.clear();
ans.push_back(i);
mini = tmp;
}
else if(tmp == mini)
ans.push_back(i);
}
printf("%d %d\n",mini,ans.size());
sort(ans.begin(),ans.end());
printf("%d",ans[]);
for(i=;i<ans.size();i++)
printf(" %d",ans[i]);
puts("");
return ;
}

树形DP 学习笔记(树形DP、树的直径、树的重心)的更多相关文章

  1. 树形DP 学习笔记

    树形DP学习笔记 ps: 本文内容与蓝书一致 树的重心 概念: 一颗树中的一个节点其最大子树的节点树最小 解法:对与每个节点求他儿子的\(size\) ,上方子树的节点个数为\(n-size_u\) ...

  2. 数位DP学习笔记

    数位DP学习笔记 什么是数位DP? 数位DP比较经典的题目是在数字Li和Ri之间求有多少个满足X性质的数,显然对于所有的题目都可以这样得到一些暴力的分数 我们称之为朴素算法: for(int i=l_ ...

  3. DP学习笔记

    DP学习笔记 可是记下来有什么用呢?我又不会 笨蛋你以后就会了 完全背包问题 先理解初始的DP方程: void solve() { for(int i=0;i<;i++) for(int j=0 ...

  4. 树形$dp$学习笔记

    今天学习了树形\(dp\),一开始浏览各大\(blog\),发现都\(TM\)是题,连个入门的\(blog\)都没有,体验极差.所以我立志要写一篇可以让初学树形\(dp\)的童鞋快速入门. 树形\(d ...

  5. [学习笔记]树形dp

    最近几天学了一下树形\(dp\) 其实早就学过了 来提高一下打开树形\(dp\)的姿势. 1.没有上司的晚会 我的人生第一道树形\(dp\),其实就是两种情况: \(dp[i][1]\)表示第i个人来 ...

  6. [总结] 动态DP学习笔记

    学习了一下动态DP 问题的来源: 给定一棵 \(n\) 个节点的树,点有点权,有 \(m\) 次修改单点点权的操作,回答每次操作之后的最大带权独立集大小. 首先一个显然的 \(O(nm)\) 的做法就 ...

  7. 洛谷4719 【模板】动态dp 学习笔记(ddp 动态dp)

    qwq大概是混乱的一个题. 首先,还是从一个比较基础的想法开始想起. 如果每次暴力修改的话,那么每次就可以暴力树形dp 令\(dp[x][0/1]\)表示\(x\)的子树中,是否选择\(x\)这个点的 ...

  8. 动态 DP 学习笔记

    不得不承认,去年提高组 D2T3 对动态 DP 起到了良好的普及效果. 动态 DP 主要用于解决一类问题.这类问题一般原本都是较为简单的树上 DP 问题,但是被套上了丧心病狂的修改点权的操作.举个例子 ...

  9. 斜率优化DP学习笔记

    先摆上学习的文章: orzzz:斜率优化dp学习 Accept:斜率优化DP 感谢dalao们的讲解,还是十分清晰的 斜率优化$DP$的本质是,通过转移的一些性质,避免枚举地得到最优转移 经典题:HD ...

随机推荐

  1. 将PDF转化为wrod

    public static void CreateWord(string HtmlPath, string WordSavePath) { string inputName = HtmlPath; / ...

  2. Scala 面向对象(二):package 包 (一) 入门

    1 Scala包的基本介绍 和Java一样,Scala中管理项目可以使用包,但Scala中的包的功能更加强大,使用也相对复杂些,下面我们学习Scala包的使用和注意事项. 2 Scala包快速入门 使 ...

  3. python 并发专题(十二):基础部分补充(四)协程

    相关概念: 协程:一个线程并发的处理任务 串行:一个线程执行一个任务,执行完毕之后,执行下一个任务 并行:多个CPU执行多个任务,4个CPU执行4个任务 并发:一个CPU执行多个任务,看起来像是同时执 ...

  4. 机器学习实战基础(十八):sklearn中的数据预处理和特征工程(十一)特征选择 之 Wrapper包装法

    Wrapper包装法 包装法也是一个特征选择和算法训练同时进行的方法,与嵌入法十分相似,它也是依赖于算法自身的选择,比如coef_属性或feature_importances_属性来完成特征选择.但不 ...

  5. redis(五):Redis 键(key)(python)

    # -*- coding: utf-8 -*- import redis #这个redis 连接不能用,请根据自己的需要修改 r =redis.Redis(host="123.516.174 ...

  6. 前端07 /jQuery初识

    前端07 /jQuery初识 目录 前端07 /jQuery初识 1.jquery介绍 1.1 jquery的优势 1.2 jquery是什么? 1.3 jquery的导入 2.jQuery的使用 2 ...

  7. OSCP Learning Notes - Exploit(5)

    Java Applet Attacks Download virtual machines from the following website: https://developer.microsof ...

  8. 将终结点图添加到你的ASP.NET Core应用程序中

    在本文中,我将展示如何使用DfaGraphWriter服务在ASP.NET Core 3.0应用程序中可视化你的终结点路由.上面文章我向您演示了如何生成一个有向图(如我上篇文章中所示),可以使用Gra ...

  9. Vue.js +pdf.js 处理响应pdf文件流数据,前端转图片预览不可下载

    使用场景及原因 实际业务中,一些说明书或协议仅支持用户在线预览,为避免用户自行下载,并进行修改,引发纠纷,特将文件已文件流的形式,传给前端并转为图片显示,此时可能会有人问,为什么不直接在后端转图片,前 ...

  10. Spring中异步注解@Async的使用、原理及使用时可能导致的问题

    前言 其实最近都在研究事务相关的内容,之所以写这么一篇文章是因为前面写了一篇关于循环依赖的文章: <面试必杀技,讲一讲Spring中的循环依赖> 然后,很多同学碰到了下面这个问题,添加了S ...