传送门

题目大意:

在一棵树中, 每条边都有一个长度值, 现要求在树中选择 3 个点 X、Y、 Z , 满足 X 到 Y 的距离不大于 X 到 Z 的距离, 且 X 到 Y 的距离与 Y 到 Z 的距离之和最大,求这个最大值。

题目分析:

一篇论文中看到了这道题,于是就来做做。

从这题中可以得到很多启示,光看题意一定会想到枚举点来做,不过如果枚举三点的话就爆了,于是枚举的那个点就成了解题的关键。我们发现,所有的答案无非就两种:

(红色代表特殊点,黄色代表起点)

他们都是最短路径 + 次短路径 * 2 + 最长路径(第一种情况相当于最短路径为0),于是第一种情况可以归到第二种情况中,这样一定是满足题意的条件下最优的。那么直接分析第二种情况:

发现无论是第一种还是第二种都有一个特殊点---分叉点A,如果用分叉点来表示距离那么|XY| + |YZ| = |XA| + 2|YA| + |ZA|,要让答案最大,也就是让三点到A的距离最大。

这下就好办了,求最大距离---树型dp:这样的三条链又有两种情况:



第一种是三条链都在子树中,第二种是一条在子树外,两条在子树中。那么进行两遍dp:第一遍从求u向下的三条不在同一颗子树的三条最长链(最长,次长,次次长),第二遍求u向上的一条最长链。

最后统计u分叉点的答案时,取这四条链的前三大,答案更新为最短路径 + 次短路径 * 2 + 最长路径

改了我一个多小时,结果发现是输出优化的int没有改成long long!!

code

#include<bits/stdc++.h>
using namespace std;
#define maxn 200050
typedef long long ll;
namespace IO{
inline int read(){
int i = 0, f = 1; char ch = getchar();
for(; (ch < '0' || ch > '9') && ch != '-'; ch = getchar());
if(ch == '-') f = -1, ch = getchar();
for(; ch >= '0' && ch <= '9'; ch = getchar())
i = (i << 3) + (i << 1) + (ch - '0');
return i * f;
}
inline void wr(ll x){
if(x < 0) putchar('-'), x = -x;
if(x > 9) wr(x / 10);
putchar(x % 10 + '0');
}
}using namespace IO;
int n, m; typedef pair<int, ll> P;
vector<P> adj[maxn];
ll dp[maxn][5], ans; inline void dfs1(int u, int f){
for(int e = adj[u].size() - 1; e >= 0; e--){
int v = adj[u][e].first;
if(v == f) continue;
ll t = adj[u][e].second;
dfs1(v, u);
if(dp[v][1] + t > dp[u][1]){
dp[u][3] = dp[u][2];
dp[u][2] = dp[u][1];
dp[u][1] = dp[v][1] + t;
}
else if(dp[v][1] + t > dp[u][2]){
dp[u][3] = dp[u][2];
dp[u][2] = dp[v][1] + t;
}
else if(dp[v][1] + t > dp[u][3]){
dp[u][3] = dp[v][1] + t;
}
}
} inline void dfs2(int u, int f){
for(int e = adj[u].size() - 1; e >= 0; e--){
int v = adj[u][e].first;
if(v == f) continue;
ll t = adj[u][e].second;
ll a = (dp[v][1] + t == dp[u][1]) ? (dp[u][2]): (dp[u][1]);
dp[v][4] = max(a + t, dp[u][4] + t);
dfs2(v, u);
}
} int main(){
n = read(), m = read();
for(int i = 1; i <= m; i++){
int x = read(), y = read(), t = read();
adj[x].push_back(P(y, 1ll*t));
adj[y].push_back(P(x, 1ll*t));
}
dfs1(1, 0); dfs2(1, 0);
for(int i = 1; i <= n; i++){
ll tmp[10];
tmp[1] = dp[i][1], tmp[2] = dp[i][2], tmp[3] = dp[i][3], tmp[4] = dp[i][4];
sort(tmp + 1, tmp + 4 + 1);
ans = max(ans, tmp[2] + tmp[3]*2 + tmp[4]);
}
wr(ans);
return 0;
}

BZOJ 1509 逃学的小孩 - 树型dp的更多相关文章

  1. BZOJ1509 [NOI2003]逃学的小孩 树型DP

    题目: 分析: 首先明确我们是要求 min(dist[C][A],dist[C][B])+dist[A][B]. 我们把C当成树根,第一我们可以发现min里面取dist[C][A]或者dist[C][ ...

  2. BZOJ 1509 逃学的小孩(树的直径)

    题意:从树上任找三点u,v,w.使得dis(u,v)+min(dis(u,w),dis(v,w))最大. 有一个结论u,v必是树上直径的两端点. 剩下的枚举w就行了. 具体不会证... # inclu ...

  3. BZOJ 1564 :[NOI2009]二叉查找树(树型DP)

    二叉查找树 [题目描述] 已知一棵特殊的二叉查找树.根据定义,该二叉查找树中每个结点的数据值都比它左儿子结点的数据值大,而比它右儿子结点的数据值小. 另一方面,这棵查找树中每个结点都有一个权值,每个结 ...

  4. BZOJ 1864 三色二叉树 - 树型dp

    传送门 题目大意: 给一颗二叉树染色红绿蓝,父亲和儿子颜色必须不同,两个儿子颜色必须不同,问最多和最少能染多少个绿色的. 题目分析: 裸的树型dp:\(dp[u][col][type]\)表示u节点染 ...

  5. [BZOJ 1509] 逃学的小孩

    Link: BZOJ 1509 传送门 Solution: 一开始受样例影响又犯了想当然的毛病……图中的C点不一定在直径上! 3次$dfs$求出树的直径及直径的两个端点$rt1,rt2$到每个点的距离 ...

  6. 【bzoj 4455】小星星(树型DP+容斥原理+dfs建树和计算的2种方式)

    题意:给一个n个点的图和一个n个点的树,求图和树上的点一一对应的方案数.(N<=17) 解法:1.在树的结构上进行tree DP,f[i][j]表示树上点 i 对应图上点 j 时,这个点所在子树 ...

  7. BZOJ 2286 消耗战 - 虚树 + 树型dp

    传送门 题目大意: 每次给出k个特殊点,回答将这些特殊点与根节点断开至少需要多少代价. 题目分析: 虚树入门 + 树型dp: 刚刚学习完虚树(好文),就来这道入门题签个到. 虚树就是将树中的一些关键点 ...

  8. POJ3659 Cell Phone Network(树上最小支配集:树型DP)

    题目求一棵树的最小支配数. 支配集,即把图的点分成两个集合,所有非支配集内的点都和支配集内的某一点相邻. 听说即使是二分图,最小支配集的求解也是还没多项式算法的.而树上求最小支配集树型DP就OK了. ...

  9. POJ 3342 - Party at Hali-Bula 树型DP+最优解唯一性判断

    好久没写树型dp了...以前都是先找到叶子节点.用队列维护来做的...这次学着vector动态数组+DFS回朔的方法..感觉思路更加的清晰... 关于题目的第一问...能邀请到的最多人数..so ea ...

随机推荐

  1. 使用差分VHD启动Win7

    平台:windows 8.1 目的:在VHD里装个Win7 介绍:建立VHD装系统姑且不表,比较有意思的是用差分VHD备份系统这点. 差分硬盘这个概念在虚拟机里常用,类似VM中的快照,不过差分硬盘更灵 ...

  2. 字符串中的空格替换问题(Java版)

    解决方式一:时间复杂度为O(n^2) 解决方式二:时间复杂度为O(n) 代码实现: package string; public class SpaceStringReplace2 { //len为数 ...

  3. jquery 获取上一个兄弟元素和下一个兄弟元素

    jQuery.prev(),返回上一个兄弟节点,不是所有的兄弟节点 jQuery.prevAll(),返回所有之前的兄弟节点 jQuery.next(),返回下一个兄弟节点,不是所有的兄弟节点 jQu ...

  4. 9、str类型和byte类型转换、列表拾遗、元组拾遗、字典拾遗、如何判断对象是否可迭代

    str(字节类型,编码)       可用于创建字符串,或者将其他的转换成字符串 a= ‘李露’ #将字符串转换成字节流 b = bytes(a,encoding = 'utf-8') #将字节转换成 ...

  5. 定制表格头, 学习Core Graphic 的第二部分, 阴影与玻璃效果.

    //定制表格头, 学习Core Graphic 的第二部分, 阴影与玻璃效果. https://github.com/comcuter/testsnippets/commit/e96f62d115ef ...

  6. 最全Pycharm教程(42)——Pycharm扩展功能之Emacs外部编辑器

    1.主题 介绍怎样将Emacs定义为一个Pycharm外部编辑器. 2.准备工作 (1)Pycharm版本号为2.7或更高 (2)下载了downloadedEmacs并正确安装 3.配置Emacs 打 ...

  7. 折叠table中的tr

    code <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF- ...

  8. 9.8 Binder系统_c++实现_内部机制1

    1. 内部机制_回顾binder框架关键点 binder进程通讯过程情景举例: test_server通过addservice向service_manager注册服务 test_client通过get ...

  9. ZOJ 2405 Specialized Four-Digit Numbers

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1405 要求找出4位数所有10进制.12进制.16进制他们各位数字之和相等. # ...

  10. HDU1248 寒冰王座 【数学题】or【全然背包】

    寒冰王座 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...