树的直径

定义

我们将一棵树T = ( V,E )的直径定义为maxδ ( u,v ) ( u,v ∈ V ),也就是说,树中所有最短路径距离的最大值即为树的直径。

寻找方法

两次bfs

方法:先从任意一点P出发,找离它最远的点Q,再从点Q出发,找离它最远的点W,W到Q的距离就是是的直径

证明如下:

①若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不一定是直线,画成直线是为了方便):

 

 --->若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是直径矛盾,所以这种情况也不成立,如下图:

 

 树状DP

对于每个节点我们要记录两个值:

f1 [ i ] 表示以 i 为根的子树中,i 到叶子结点距离的最大值

f2 [ i ] 表示以 i 为根的子树中,i 到叶子结点距离的次大值

对于一个节点,它到叶子结点距离的最大值和次大致所经过的路径肯定是不一样的

若j是i的儿子,那么(下面的 w [ i ][ j ] 表示 i 到 j 的路径长度):

若 f1 [ i ] < f1 [ j ] + w [ i ][ j ],f2 [ i ] = f1 [ i ],f1 [ i ] = f1 [ j ] + w [ i ][ j ];否则,若 f2 [ i ] < f1 [ j ] + w [ i ][ j ],f2 [ i ] = f1 [ j ] + w [ i ][ j ]; 

最后的答案 answer = max { f1 [ i ] + f2 [ i ] }

 

习题

https://www.luogu.com.cn/problem/P5536

 

 

#include<iostream>
#include<cstdio>
#include<string>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
int n, k,cnt=0,mdeep=-1,mi;
#define maxn 100010
struct edge
{
    int to;
    int next;
}e[maxn *2];
int father[maxn],head[maxn],deep[maxn],maxdeep[maxn],dis[maxn];
bool cmp(int a, int b)
{
    return a > b;
}
void addedge(int u,int v)
{
    cnt++;
    e[cnt].to = v;
    e[cnt].next = head[u];
    head[u] = cnt;
}
void dfs1(int s,int fa)
{
    if (deep[s] > mdeep)
    {
        mdeep = deep[s];
        mi = s;
    }

    for (int i = head[s]; i; i = e[i].next)
    {
        int y = e[i].to;
        if (y == fa)continue;
        deep[y] = deep[s] + 1;
        dfs1(y, s);
    }
}

void dfs2(int s, int fa)
{
    if (deep[s] > mdeep)
    {
        mdeep = deep[s];
        mi = s;
    }

    for (int i = head[s]; i; i = e[i].next)
    {
        int y = e[i].to;
        if (y == fa)continue;
        father[y] = s;//记录路径
        deep[y] = deep[s] + 1;
        dfs2(y, s);
    }
}
void dfsk(int s, int fa)
{
    maxdeep[s] = deep[s];
    for (int i = head[s]; i; i = e[i].next)
    {
        int y = e[i].to;
        if (y == fa)continue;
        deep[y] = deep[s] + 1;
        dfsk(y, s);
        maxdeep[s] = max(maxdeep[s], maxdeep[y]);//自己的子节点能到达的最深深度,自己也一定能去
        
    }
}
int main()
{
    scanf("%d%d", &n, &k);
    for (int i = 1; i <=n - 1; i++)
    {
        int a, b;
        scanf("%d%d", &a, &b);
        addedge(a, b);
        addedge(b, a);
    }
    dfs1(1, 0);
    memset(deep, 0, sizeof(deep));
    mdeep = -1;
    dfs2(mi, 0);
    int temp = mi;//mi:直径末端上的点
    for (int i = 1; i <= (deep[mi]+1) / 2; i++)temp = father[temp];//之前的记录路径就是为了这里服务的,从直径的末端走该直径长度的一半就是直径的中点
    memset(deep, 0, sizeof(deep));
    dfsk(temp, 0);
    for (int i = 1; i <= n; i++)
        dis[i] = maxdeep[i] - deep[i];
    sort(dis + 1, dis + n + 1, cmp);
    int ans = 0;
    for (int i = k + 1; i <= n; i++)
        ans = max(ans, dis[i] + 1);
    printf("%d\n", ans);
}

参考:https://blog.csdn.net/forever_dreams/article/details/81051578

posted @ 2020-06-20 15:23  Jason66661010  阅读(325)  评论(0编辑  收藏  举报