换根dp就是先任取一点为根,预处理出一些信息,然后在第二次dfs过程中进行状态的转移处理

本题难点在于任意割断一条边,求出剩下两棵子树的直径:

  设割断的边为(u,v),设down[v]为以v为根的子树的直径长度,up[v]为u所在的子树的直径长度,那么down[v]就是很常规的子树直径的换根dp的求法,up[v]则要通过分情况讨论

  第一种情况,组成up[v]的两条链,一条是u上方的链,一条是u下方且不属于子树v的链

  第二种情况,组成up[v]的两条链都是u下方且不属于子树v的链

那么换根的过程中就要考虑这两种情况,我们必须维护u下的前三条最长的链,和u上一条最长的链,

因为如果v刚好在u深度最大的子树里,考虑第二种情况的up[v]时就要用到u下次长,次次长的链,而考虑第一种情况时就必须要用到u上的最长链,所以最少需要维护u下三条链+u上一条链

切断任意一条边,求剩下两个子树的直径
换根dp,第一次dfs预先处理出的数据dp[u][0|1|2]表示u的最长|次长|次次长 链
down[u]表示u子树里的最长链长度

第二次dfs求出
len[u][0|1]表示u子树里的不经过u的最长|次长 链
枚举每个儿子v,
求出dp[v][3]表示v上面的最长链
up[v]表示切断(u,v)后,u所在块的最长链
那么切断(u,v)后,两个子树的直径就是down[v],up[v]
down[v]好求,up[v]要通过u来求出

换根的过程:从u换到v时,up[v]有两种情况,一种是一条u的上面+一条u的下面,另一种是两条u的下面

#include<bits/stdc++.h>
#include<vector>
using namespace std;
#define N 200005
vector<int>G[N];
int n,u[N],v[N],a[N],down[N],up[N],dp[N][],len[N][],d[N];
void dfs1(int u,int pre,int dep){
d[u]=dep;
for(int i=;i<G[u].size();i++){
int v=G[u][i];
if(v==pre)continue;
dfs1(v,u,dep+);
int tmp=dp[v][]+;
if(tmp>dp[u][])swap(dp[u][],tmp);
if(tmp>dp[u][])swap(dp[u][],tmp);
if(tmp>dp[u][])swap(dp[u][],tmp);
down[u]=max(down[u],down[v]);
}
down[u]=max(down[u],dp[u][]+dp[u][]);
}
void dfs2(int u,int pre){
for(int i=;i<G[u].size();i++){//求出u下不经过u的最长|次长 链
int v=G[u][i];
if(v==pre)continue;
int tmp=down[v];
if(tmp>len[u][])swap(tmp,len[u][]);
if(tmp>len[u][])swap(tmp,len[u][]);
}
for(int i=;i<G[u].size();i++){//边(u,v)将原树分成两棵子树
int v=G[u][i];
if(v==pre)continue;
//原树删掉v子树后,求v上的最长链dp[v][3],同时求出第一种情况的up[v]
if(dp[u][]==dp[v][]+){//v是u的最深子树
dp[v][]=max(dp[u][],dp[u][])+;
up[v]=max(dp[u][],dp[u][])+dp[u][];
}
else if(dp[u][]==dp[v][]+){//v是u的次深子树
dp[v][]=max(dp[u][],dp[u][])+;
up[v]=max(dp[u][],dp[u][])+dp[u][];
}
else {//v是u的其他子树
dp[v][]=max(dp[u][],dp[u][])+;
up[v]=max(dp[u][],dp[u][])+dp[u][];
}
//求第二种情况的up[v],也要特别判一下v子树里是否有u下的最长链
if(len[u][]==down[v])up[v]=max(up[v],len[u][]);
else up[v]=max(up[v],len[u][]);
dfs2(v,u);
}
}
void init(){
memset(dp,,sizeof dp);
memset(a,,sizeof a);
memset(len,,sizeof len);
memset(down,,sizeof down);
memset(up,,sizeof up);
for(int i=;i<=n;i++)G[i].clear();
}
int main(){
int t;cin>>t;while(t--){
init();cin>>n;
for(int i=;i<n;i++){
scanf("%d%d",&u[i],&v[i]);
G[u[i]].push_back(v[i]);
G[v[i]].push_back(u[i]);
}
dfs1(,,);
dfs2(,);
for(int i=;i<n;i++){
int x=u[i],y=v[i];
if(d[x]<d[y])swap(x,y);
a[up[x]+]=max(a[up[x]+],down[x]+);
a[down[x]+]=max(a[down[x]+],up[x]+);
}
long long ans=;
for(int i=n;i>=;i--){
a[i]=max(a[i],a[i+]);
ans+=a[i];
}
cout<<ans<<'\n';
}
}

树形dp换根,求切断任意边形成的两个子树的直径——hdu6686的更多相关文章

  1. bzoj 3743 [Coci2015]Kamp——树形dp+换根

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3743 树形dp+换根. “从根出发又回到根” 减去 “mx ” . 注意dfsx里真的要改那 ...

  2. poj3585 Accumulation Degree(树形dp,换根)

    题意: 给你一棵n个顶点的树,有n-1条边,每一条边有一个容量z,表示x点到y点最多能通过z容量的水. 你可以任意选择一个点,然后从这个点倒水,然后水会经过一些边流到叶节点从而流出.问你最多你能倒多少 ...

  3. poj3585 Accumulation Degree[树形DP换根]

    思路其实非常简单,借用一下最大流求法即可...默认以1为根时,$f[x]$表示以$x$为根的子树最大流.转移的话分两种情况,一种由叶子转移,一种由正常孩子转移,判断一下即可.换根的时候由頂向下递推转移 ...

  4. [题解](树形dp/换根)小x游世界树

    2. 小x游世界树 (yggdrasi.pas/c/cpp) [问题描述] 小x得到了一个(不可靠的)小道消息,传说中的神岛阿瓦隆在格陵兰海的某处,据说那里埋藏着亚瑟王的宝藏,这引起了小x的好奇,但当 ...

  5. Acwing-287-积蓄程度(树上DP, 换根)

    链接: https://www.acwing.com/problem/content/289/ 题意: 有一个树形的水系,由 N-1 条河道和 N 个交叉点组成. 我们可以把交叉点看作树中的节点,编号 ...

  6. poj3162(树形dp+线段树求最大最小值)

    题目链接:https://vjudge.net/problem/POJ-3162 题意:给一棵树,求每个结点的树上最远距离,记为a[i],然后求最大区间[l,r]满足区间内的max(a[i])-min ...

  7. 2019ICPC沈阳网络赛-D-Fish eating fruit(树上DP, 换根, 点分治)

    链接: https://nanti.jisuanke.com/t/41403 题意: State Z is a underwater kingdom of the Atlantic Ocean. Th ...

  8. bzoj 1131 [POI2008]Sta 树形dp 转移根模板题

    [POI2008]Sta Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1889  Solved: 729[Submit][Status][Discu ...

  9. cf219d 基础换根法

    /*树形dp换根法*/ #include<bits/stdc++.h> using namespace std; #define maxn 200005 ]; int root,n,s,t ...

随机推荐

  1. secureCRT The remote system refused the connection.解决办法

    使用远程登录工具SecureCRT登陆ubuntu的时候遇到了这个问题: secureCRT The remote system refused the connection 这个问题的原因是是Ubu ...

  2. 说一下HTML5与传统HTML的区别

    1.首先说一下什么是HTML5 HTML5 将成为 HTML.XHTML 以及 HTML DOM 的新标准. HTML 的上一个版本诞生于 1999 年.自从那以后,Web 世界已经经历了巨变. HT ...

  3. redis集群添加新节点

    一.创建节点(接上文) 1.在H1服务器/root/soft目录下创建7002目录 2.将7001目录的配置文件redis.conf拷贝到7002,并修改配置文件的端口 3.进入 redis-5.0. ...

  4. thinkphp IP获取和定位

    系统内置了get_client_ip方法用于获取客户端的IP地址,使用示例: $ip = get_client_ip(); 如果要支持IP定位功能,需要使用扩展类库Org\Net\IpLocation ...

  5. P1493 分梨子

    P1493 分梨子 题目描述 Finley家的院子里有棵梨树,最近收获了许多梨子.于是,Finley决定挑出一些梨子,分给幼稚园的宝宝们.可是梨子大小味道都不太一样,一定要尽量挑选那些差不多的梨子分给 ...

  6. PHP面试 PHP基础知识 五(自定义函数和内部函数)

    自定义函数 变量的作用域和静态变量 变量的作用域:变量的作用域也成为变量的范围,变量的范围即它定义上的上下文背景(也就是它生效的范围). 大部分的PHP变量只有一个单独的范围.这个单独的范围跨度同样包 ...

  7. 剑指offer——46数字序列中某一位的数字

    题目: 数字以0123456789101112131415…的格式序列化到一个字符序列中.在这个序列中,第5位(从0开始计数)是5,第13位是1,第19位是4,等等.请写一个函数,求任意第n位对应的数 ...

  8. Java的核心优势

    Java为消费类智能电子产品而设计,但智能家电产品并没有像最初想象的那样拥有大的发展.然而90年代,Internet却进入了爆发式发展阶段,一夜之间,大家都在忙着将自己的计算机连接到网络上.这个时侯, ...

  9. cm 安装cdh 后添加hive服务

    cm 安装cdh 后添加hive服务,出现错误提示 添加服务时候hive 配置如下: 错误信息提示: 错误日志: xec /opt/cloudera/parcels/CDH-5.4.7-1.cdh5. ...

  10. python实现马赛克拼图!

    python实现马赛克拼图 直接上代码! 代码如下: #!/usr/local/bin/python3#  --*-- coding:utf8 --*-- import getoptimport sy ...