题目大意
给出一棵树,其中每两个节点都可以形成一个路径(要求路径中的边只能走一次),求出所有路径中的长度最大值。
分析
树形结构,很容易想到递归,但为了节省时间,要考虑保存中间状态。于是,考虑使用记忆化搜索(也就是树形动态规划)。
保存状态 dp[i][2],其中dp[i][0]表示以i为根的子树中路径的两个端点均不位于i的路径的最长值,dp[i][1]表示以i为根的子树中有一个端点位于i的路径的最长值。然后进行状态推演,
dp[root][1] = 1 + max(dp[child][1]);
dp[root][0] = max(max(max0, 2 + max1 + max2);(root的子节点数大于1)
dp[root][0] = 1 + max1;(root的子节点数等于1)
max0表示i的所有子节点中的最大的dp[c][0], max1表示i的所有字节点中最大的dp[c][1], max2表示i的所有子节点中第二大的dp[c][1].
由于树的任何一个节点均可以作为根节点,因此dfs时候,选择1即可。
实现
#pragma once #pragma execution_character_set("utf-8") // 本文件为utf-8 编码格式 #include<iostream> #include<stdio.h> #include<string.h> using namespace std; #define N 100005 int dp[N][2]; struct Edge{ int to; int next; }; Edge gEdges[2*N]; int gHead[N]; bool gVisited[N]; int gEdgeIndex; void InsertEdge(int u, int v){ int e = gEdgeIndex++; gEdges[e].to = v; gEdges[e].next = gHead[u]; gHead[u] = e; e = gEdgeIndex++; gEdges[e].to = u; gEdges[e].next = gHead[v]; gHead[v] = e; } pair<int, int> dfs(int root){ if (dp[root][0] != -1 && dp[root][1] != -1){ return pair<int, int>(dp[root][0], dp[root][1]); } gVisited[root] = true; int e = gHead[root]; int max1 = 0, max2 = 0, max = 0, child_num = 0; for (; e != -1; e = gEdges[e].next){ int v = gEdges[e].to; if (!gVisited[v]){ pair<int, int> result = dfs(v); max = max > result.first ? max : result.first; if (max1 >= result.second){ max2 = max2 > result.second ? max2 : result.second; } else{//求一组数中的第一大和第二大的数!!! 注意次序 /* //这样做,在处理第一个result的时候, max1和max2赋值为同一个...error max1 = result.second; max2 = max1; //这样做,考虑到第一个值的处理,但是 对max1和max2的更新次序错了。 仔细考虑... max1 = result.second; if(max1 != 0) max2 = max1; */ if (max1 != 0) max2 = max1; max1 = result.second; } child_num++; } } if (child_num == 0) dp[root][0] = dp[root][1] = 0; else if (child_num == 1){ dp[root][0] = max; dp[root][1] = 1 + max1; } else{ dp[root][1] = 1 + max1; dp[root][0] = max > (2 + max1 + max2) ? max : (2 + max1 + max2); } return pair<int, int>(dp[root][0], dp[root][1]); } void Init(){ memset(gVisited, false, sizeof(gVisited)); memset(gHead, -1, sizeof(gHead)); gEdgeIndex = 0; memset(gEdges, -1, sizeof(gEdges)); memset(dp, -1, sizeof(dp)); } int main(){ int n, u, v; scanf("%d", &n); Init(); for (int i = 1; i < n; i++){ scanf("%d %d", &u, &v); InsertEdge(u, v); } pair<int, int> result = dfs(1); printf("%d\n", result.first>result.second ? result.first : result.second); return 0; }
时间: 2024-10-26 20:05:57