树的直径算法

树上任意两节点之间最长的简单路径即为树的「直径」。记 \(\delta(s, t)\) 为树的真实直径。

算法1,两遍dfs

首先从任意节点 \(y\) 开始进行第一次 DFS,到达距离其最远的节点,记为 \(z\),然后再从 \(z\) 开始做第二次 DFS,到达距离 \(z\) 最远的节点,记为 \(z'\),则 即为树的直径。

那么为什么此算法可行?

我们只需证明对于任意一点 \(x\) 经过第一次dfs之后到达的点 \(y\) 一定是 \(\delta(s,t)\) 的一端 \(s\)\(t\)

定理:在一棵树上,从任意节点 \(x\) 开始进行一次 DFS,到达的距离其最远的节点 \(z\) 必为直径的一端。

我们采用反证法。记出发点为 \(y\),真实直径 \(\delta(s,t)\),设从 \(y\) 进行的第一次dfs到达的距离其最远的点 \(z\) 不为 \(t\)\(s\)
那么有三种情况:

  • \(y\)\(\delta(s,t)\) 上:那么 \(\delta(y,z)>\delta(y,t)\Rightarrow \delta(x,z)>\delta(x,t)\Rightarrow \delta(s,z)>\delta(s,t)\) 矛盾。其中 \(x\)\(\delta(s,t)\) 上一中转点
    image
  • \(y\) 不在直径上,且 \(\delta(y,z)\)\(\delta(s,t)\) 有重合路径
    image
    那么 \(\delta(y,z)>\delta(y,t)\Rightarrow \delta(x,z)>\delta(x,t)\Rightarrow \delta(s,z)>\delta(s,,t)\) 矛盾。
  • \(y\) 不在 \(\delta(s,t)\) 上,且 \(\delta(y,z)\)\(\delta(s,t)\) 不存在重复路径:
    image
    那么 \(\delta(y,z)>\delta(y,t)\Rightarrow \delta(x',z)>\delta(x',t)\Rightarrow\delta(x,z)>\delta(x,t)\Rightarrow\delta(s,z)>\delta(s,t)\)矛盾。
  • 三种情况都矛盾,得证。

注意负权边不适用

那么代码实现就很简单了

const int N = 10005;

int n, c, d[N];
vector<int> e[N];

void dfs(int u, int fa) {
  for (int v :e[u]) {
    if (v == fa) continue;
    d[v] = d[u] + 1;
    if (d[v] > d[c]) c = v;
    dfs(v, u);
  }
}

int main() {
  scanf("%d", &n);
  for (int i = 1; i < n; i++) {
    int u, v;
    scanf("%d %d", &u, &v);
    e[u].push_back(v), e[v].push_back(u);
  }
  dfs(1, 0);
  d[c] = 0, dfs(c, 0);
  printf("%d\n", d[c]);
  return 0;
}

\(O(N)\)?

树形dp

每个节点作为子树的根向下,所能延伸的最远距离 \(d1\),和次远距离 \(d2\),那么直径就是所有 \(d1+d2\) 的最大值。

树形 DP 可以在存在负权边的情况下求解出树的直径。

const int N=10010,M=20010;
int n,a,b,c,ans;
struct edge{int v,w;};
vector<edge> e[N];

int dfs(int x,int fa){
  int d1=0,d2=0; 
  for(auto ed : e[x]){
    int y=ed.v, z=ed.w;
    if(y==fa) continue;
    int d=dfs(y,x)+z;
    if(d>=d1) d2=d1,d1=d;
    else if(d>d2) d2=d;
    // printf("回%d d1=%d d2=%d\n",x,d1,d2);
  }
  ans=max(ans,d1+d2);
  // printf("离%d ans=%d\n",x,ans);
  return d1;
}
int main(){
  cin>>n;
  for(int i=1; i<n; i++){
    cin>>a>>b>>c;
    e[a].push_back({b,c});
    e[b].push_back({a,c});
  }
  dfs(1,-1);
  cout << ans << endl;
  return 0;
}

最后的最后,始终要记住,树是无环的,这一性质很重要,在两个算法中都用到了。

posted @ 2022-12-26 13:14  Vegdie  阅读(337)  评论(0编辑  收藏  举报