这是找树的重心的经典题目。
树的重心有下面几条常见性质:
定义1:找到一个点,其所有的子树中最大的子树节点数最少,那么这个点就是这棵树的重心。
定义2:以这个点为根,那么所有的子树(不算整个树自身)的大小都不超过整个树大小的一半。
性质1:树中所有点到某个点的距离和中,到重心的距离和是最小的;如果有两个重心,那么他们的距离和一样。
性质2:把两个树通过一条边相连得到一个新的树,那么新的树的重心在连接原来两个树的重心的路径上。
性质3:把一个树添加或删除一个叶子,那么它的重心最多只移动一条边的距离。
dfs一一遍求所有的子树的size
void dfssize(int u, int fa) //处理子树的大小
{
size[u] = 1;
mx[u] = 0;
for(int i = head[u]; i != -1; i = edge[i].next)
{
int v = edge[i].v;
if(v != fa && !vis[v])
{
dfssize(v, u);
size[u] += size[v];
if(size[v] > mx[u]) mx[u] = size[v];
}
}
}
{
size[u] = 1;
mx[u] = 0;
for(int i = head[u]; i != -1; i = edge[i].next)
{
int v = edge[i].v;
if(v != fa && !vis[v])
{
dfssize(v, u);
size[u] += size[v];
if(size[v] > mx[u]) mx[u] = size[v];
}
}
}
求最大子树size并且这个最大是所有最大的最小那一个
void dfsroot(int r, int u, int fa) //求重心
{
if(size[r] - size[u] > mx[u]) mx[u] = size[r] - size[u];
if(mx[u] < mi) mi = mx[u], root = u;
for(int i = head[u]; i != -1; i = edge[i].next)
{
int v = edge[i].v;
if(v != fa && !vis[v]) dfsroot(r, v, u);
}
}
void dfsroot(int r, int u, int fa) //求重心
{
if(size[r] - size[u] > mx[u]) mx[u] = size[r] - size[u];
if(mx[u] < mi) mi = mx[u], root = u;
for(int i = head[u]; i != -1; i = edge[i].next)
{
int v = edge[i].v;
if(v != fa && !vis[v]) dfsroot(r, v, u);
}
}