树的重心

定义:

树的重心也叫树的质心。找到一个点,其所有的子树中最大的子树节点数最少,那么这个点就是这棵树的重心,删去重心后,生成的多棵树尽可能平衡。

性质:

1. 树中所有点到某个点的距离和中,到重心的距离和是最小的,如果有两个重心,他们的距离和一样。
2. 把两棵树通过一条边相连,新的树的重心在原来两棵树重心的连线上。
3. 一棵树添加或者删除一个节点,树的重心最多只移动一条边的位置。
4. 一棵树最多有两个重心,且相邻。

求法:

 定义dp[i]表示以 i 为根节点的子树的大小。若节点 i 的子节点有v1、v2、v3···vn,则dp[i] = dp[v1] + dp[v2] + dp[v3] + ··· + dp[vn]。
 对于一个节点 i , 如果我们把它从树上删除 ,那么原来的一棵树可能会被分成若干个不相连的部分,其中每一部分都是一颗子树。
 设 maxs(i) 表示 删除节点 i 后 产生的子树中,最大的一棵的大小. 使 maxs(i) 取到最小值的节点 i 就称为 整棵树的重心.

//链式前向星存图

void dfs(int u, int fa){
	dp[u] = 1;	//子树u的大小
	int maxs = 0;
	for(int i = head[u]; ~i; i = e[i].next){
		int v = e[i].v;
		if(v != fa){
			dfs(v, u);
			dp[u] += dp[v];
			maxs = max(maxs, dp[v]);	// 找到产生的所有子树中节点数最大的
		}
	}
	maxs = max(maxs, n - dp[u]);
	if(maxs < ans){
		ans = maxst ;// 重心对应的 max_part值
		r2 = r1, r1 = u;
	}
/*
	if((maxs << 1) <= n)	
		r2 = r1, r1 = u;
*/
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值