树的直径&树的重心
树的直径
定义
那么树上最远的两个点,他们之间的距离,就被称之为树的直径。
树的直径的性质
1. 直径两端点一定是两个叶子节点。
2. 距离任意点最远的点一定是直径的一个端点,这个基于贪心求直径方法的正确性 可以得出。
3. 对于两棵树,如果第一棵树直径两端点为(u,v),第二棵树直径两端点为 (x,y),用条边将两棵树连接,那么新树的直径一定是u,v,x,y中的两个点。
4. 对于一棵树,如果在一个点上接一个叶子节点,那么最多会改变直径的一个端 点。
5. 若一棵树存在多条直径,那么这些直径交于一点且交点是这些直径的中点。
树的直径的求法
解法1:树型DP
直接上代码:
void dp(int x,int fa){//f[i]表示以i为根节点的最长链
for(int i=head[x];i;i=e[i].next){
int y=e[i].v;
if(y==fa) continue;//防止死循环
dp(y,x);
ans=max(ans,f[x]+f[y]+e[i].w);//更新最长链+次长链的长度
f[x]=max(f[x],f[y]+e[i].w);//更新f[x]的长度
}
}
树形DP有一些难理解,但他可以解决负边权的问题
解法2:两次dfs/bfs
上代码:
void dfs(int x,int fa){//dis[i]表示以i的最长距离
if(dis[x]>ans){
ans=dis[x];//更新
p=x;//记录最远的点,因为要两次dfs才能确定直径
}
for(int i=head[x];i;i=e[i].next){
int y=e[i].v;
if(y==fa)continue;
dis[y]=dis[x]+e[i].w;//计算最长距离
dfs(y,x);
}
}
解法2运用了性质2,易理解,但不能处理负边权问题。
上一道例题:https://www.luogu.com.cn/problem/P3629
AC代码:
#include<bits/stdc++.h>
using namespace std;
const int N=1e5+5;
int n,k;
int a,b;
int p;
int idx=1;
int head[N];
int dis[N];
int pass[N];
int f[N];
int ans=0; struct node{
int v;
int next;
int w;
}e[N*2]; void add(int a,int b){
e[++idx].v=b;
e[idx].next=head[a];
e[idx].w=1;
head[a]=idx;
} void dfs(int x,int fa){
if(dis[x]>ans){
ans=dis[x];
p=x;
}
for(int i=head[x];i;i=e[i].next){
int y=e[i].v;
if(y==fa)continue;
dis[y]=dis[x]+e[i].w;
pass[y]=i;
dfs(y,x);
}
} void dp(int x,int fa){
for(int i=head[x];i;i=e[i].next){
int y=e[i].v;
if(y==fa) continue;
dp(y,x);
ans=max(ans,f[x]+f[y]+e[i].w);
f[x]=max(f[x],f[y]+e[i].w);
}
} int main(){
cin>>n>>k;
for(int i=1;i<n;i++){
cin>>a>>b;
add(a,b);
add(b,a);
}
dfs(1,0);
memset(dis,0,sizeof(dis));
memset(pass,0,sizeof(pass));
ans=0;
dfs(p,0);
if(k==1){
cout<<2*(n-1)-ans+1;
return 0;
}
int l1=ans;
ans=0;
while(pass[p]){
e[pass[p]].w=-1;
e[pass[p]^1].w=-1;
p=e[pass[p]^1].v;
}
dp(1,0);
cout<<2*(n-1)-l1-ans+2;
return 0;
}
树的重心
定义
树的重心也叫树的质心。对于一棵 个节点的无根树,找到一个点,使得把树变成以该 点为根的有根树时,最大子树的结点数最小。换句话说,删除这个点后最大连通块(一定是 树)的结点数最小,删去重心后,生成的多棵树尽可能平衡。
性质
1. 树中所有点到某个点的距离和中,到重心的距离和是最小的,如果有两个重心, 他们的距离和一样。
2. 把两棵树通过一条边相连,新的树的重心在原来两棵树重心的连线上。
3. 一棵树添加或者删除一个节点,树的重心最多只移动一条边的位置。
4. 一棵树最多有两个重心,且相邻
思路
找到以i为根结点的最大子树大小,与现在最小最大子树相比较,不断更新找出重心
//minNode当前重心节点
//minBalance当前重心节点的最大子树节点个数
int d[maxn];
//d[i]表示以i为根的子树节点个数
void dfs(int u,int fa){
d[u]=1; //节点本身
int maxSub=0,size=tree[u].size(); //maxSub为节点u的最大子树节点个数
for(int i=0;i<size;i++){
int v=tree[u][i];
if(v!=fa){
dfs(v,u);
d[u]+=d[v];
maxSub=max(maxSub,d[v]);
}
}
maxSub=max(maxSub,n-d[u]);
if(maxSub<minBalance){
minNode=u;
minBalance=maxSub;
}
}
树的直径&树的重心的更多相关文章
- 【SDOI2011 第2轮 DAY1】消防 -[树的直径+树链剖分][解题报告]
[SDOI2011 第2轮 DAY1]消防 题面: SDOI2011 第2轮 DAY1]消防 时间限制 : 20000 MS 空间限制 : 565536 KB 问题描述 时限\(2s\) 某个国家有\ ...
- 树形DP 学习笔记(树形DP、树的直径、树的重心)
前言:寒假讲过树形DP,这次再复习一下. -------------- 基本的树形DP 实现形式 树形DP的主要实现形式是$dfs$.这是因为树的特殊结构决定的——只有确定了儿子,才能决定父亲.划分阶 ...
- D4 树的直径、重心以及基环树
第一题第二题鉴上我前几篇博客poj1985 poj1849:https://www.cnblogs.com/Tyouchie/p/10384379.html 第三题:数的重心:poj1655 来自sj ...
- Codeforces 1182D Complete Mirror 树的重心乱搞 / 树的直径 / 拓扑排序
题意:给你一颗树,问这颗树是否存在一个根,使得对于任意两点,如果它们到根的距离相同,那么它们的度必须相等. 思路1:树的重心乱搞 根据样例发现,树的重心可能是答案,所以我们可以先判断一下树的重心可不可 ...
- POJ 树的直径和重心
树的直径:(无根)树上最长两点间的最长路径,两次dfs即可,第一次dfs任选一点u,找到距离它最远的点s,再从点s进行一次dfs,找到距离s最远的点t,则s-t之间的路径就是树的直径.证明: < ...
- 算法笔记--树的直径 && 树形dp && 虚树 && 树分治 && 树上差分 && 树链剖分
树的直径: 利用了树的直径的一个性质:距某个点最远的叶子节点一定是树的某一条直径的端点. 先从任意一顶点a出发,bfs找到离它最远的一个叶子顶点b,然后再从b出发bfs找到离b最远的顶点c,那么b和c ...
- 换根DP+树的直径【洛谷P3761】 [TJOI2017]城市
P3761 [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公 ...
- [TJOI2017]城市 【树的直径+暴力+优化】
Online Judge:Luogu P3761 Label:树的直径,暴力 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有n座城市,n-1条高速公路,保证了 ...
- poj2631 求树的直径裸题
题目链接:http://poj.org/problem?id=2631 题意:给出一棵树的两边结点以及权重,就这条路上的最长路. 思路:求实求树的直径. 这里给出树的直径的证明: 主要是利用了反证法: ...
随机推荐
- python的默认参数的一个坑
前言 pass 正文 在 https://docs.python.org/3/tutorial/controlflow.html#default-argument-values 中,有这样一段话 Im ...
- 剑指offer 面试题9.1:用两个队列实现栈
题目描述 使用队列实现栈的下列操作:push(x) -- 元素 x 入栈:pop() -- 移除栈顶元素:top() -- 获取栈顶元素:empty() -- 返回栈是否为空: 编程思想 利用双队列实 ...
- 基于Python的接口自动化-读写excel文件
引言 使用python进行接口测试时常常需要接口用例测试数据.断言接口功能.验证接口响应状态等,如果大量的接口测试用例脚本都将接口测试用例数据写在脚本文件中,这样写出来整个接口测试用例脚本代码将看起来 ...
- LeetCode53 最大子序列问题
题目描述: 给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. 示例: 输入: [-2,1,-3,4,-1,2,1,-5,4], ...
- 谈谈你不知道的gist
1.Gist是什么关于Gist的详细介绍,请阅读官方文档About gists,下面只简略介绍部分功能: Gist是一种与其他人共享代码片段和粘贴的简单方法. 当您需要与同事或朋友共享示例代码或技术时 ...
- 【Spring】Spring的数据库开发 - 2、Spring JdbcTemplate的常用方法(execute、update、query)
Spring JdbcTemplate的常用方法 文章目录 Spring JdbcTemplate的常用方法 execute() update() query() 简单记录-Java EE企业级应用开 ...
- MySQL全面瓦解17:触发器相关
关于触发器 现实开发中我们经常会遇到这种情况,比如添加.删除和修改信息的时候需要记录日志,我们就要在完成常规的数据库逻辑操作之后再去写入日志表,这样变成了两步操作,更复杂了. 又比如删除一个人员信息的 ...
- 在Jetbrain IDE中自定义TODO功能
好的IDE能为开发以及学习源码带来效率的提升,今天要介绍的就是Jetbrain家族中IDE自带的TODO功能,我认为利用好它,能够大大的提升阅读源码的效率. 假设我现在需要去阅读源代码,看了半天我终于 ...
- Genymotion虚拟机用键盘输入中文
genymotion我用的版本是3.0.4,安卓内核版本从4到9都进行了尝试,尤其是教新的版本原生是不带中文输入法的. 前提:安装Genymotion以后,想要随意安装app,需要先安装Genymot ...
- 微服务网关1-Spring Cloud Gateway简介
一.网关基本概念 1.API网关介绍 API 网关出现的原因是微服务架构的出现,不同的微服务一般会有不同的网络地址,而外部客户端可能需要调用多个服务的接口才能完成一个业务需求,如果让客户端直接与各 ...