P4408 逃学的小孩 题解
题目描述

Chris家的电话铃响起了,里面传出了Chris的老师焦急的声音:“喂,是Chris的家长吗?你们的孩子又没来上课,不想参加考试了吗?”一听说要考试,Chris的父母就心急如焚,他们决定在尽量短的时间内找到Chris。他们告诉Chris的老师:“根据以往的经验,Chris现在必然躲在朋友Shermie或Yashiro家里偷玩《拳皇》游戏。现在,我们就从家出发去找Chris,一但找到,我们立刻给您打电话。”说完砰的一声把电话挂了。
Chris居住的城市由N个居住点和若干条连接居住点的双向街道组成,经过街道x需花费Tx分钟。可以保证,任两个居住点间有且仅有一条通路。Chris家在点C,Shermie和Yashiro分别住在点A和点B。Chris的老师和Chris的父母都有城市地图,但Chris的父母知道点A、B、C的具体位置而Chris的老师不知。
为了尽快找到Chris,Chris的父母会遵守以下两条规则:
如果A距离C比B距离C近,那么Chris的父母先去Shermie家寻找Chris,如果找不到,Chris的父母再去Yashiro家;反之亦然。
Chris的父母总沿着两点间唯一的通路行走。
显然,Chris的老师知道Chris的父母在寻找Chris的过程中会遵守以上两条规则,但由于他并不知道A,B,C的具体位置,所以现在他希望你告诉他,最坏情况下Chris的父母要耗费多长时间才能找到Chris?
输入格式
输入文件第一行是两个整数\(N(3 ≤ N ≤ 200000)\)和\(M\),分别表示居住点总数和街道总数。
以下M行,每行给出一条街道的信息。第\(i+1\)行包含整数\(U_i、V_i、T_i(1≤U_i, V_i ≤ N,1 ≤ T_i ≤ 1000000000)\),表示街道i连接居住点\(U_i\)和\(V_i\),并且经过街道i需花费\(T_i\)分钟。街道信息不会重复给出。
输出格式
输出文件仅包含整数T,即最坏情况下Chris的父母需要花费T分钟才能找到Chris。
输入样例
4 3
1 2 1
2 3 1
3 4 1
输出样例
4
题解
好久没见过思路这么简单的题了,但该WA还是要WA
首先这个图是一棵树
然后就很明确了,显然是树的直径,情况最差的时候,A点和B点之间的路径就是树的直径,然后C点就是使\(min(distance_{C,A},distance_{C,B})\)最大的C点.
首先DFS求出树的直径,然后遍历每一个点作为C点求出\(distance_{C,A},distance_{C,B}\),然后根据刚才的式子推出来即可.
有几个需要注意的点:
9 8
1 8 1000000000
8 7 1000000000
7 5 1000000000
1 3 1000000000
7 2 1000000000
1 6 1000000000
3 4 1000000000
8 9 1000000000
对于这个输入,应该输出8000000000,如果你输出的是负数说明你没有开long long.
直接些会TLE,在求距离的函数那里加一个记忆化即可
怎么感觉我写不是正解,代码也太丑了
再附赠一个数据:
输入:
4 3
1 2 1
1 3 1
1 4 2
输出:
5
代码
#include <cstdio>
#include <iostream>
#include <map>
#include <utility>
#define P pair<int, int>
using namespace std;
const int maxn = 200005;
int head[maxn], tot, maxi, n, m, ix, iy, iv;
long long maxv;
struct Edge {
int to, next;
long long value;
} edges[maxn << 1];
void add(int x, int y, long long value) {
edges[++tot].to = y;
edges[tot].next = head[x];
edges[tot].value = value;
head[x] = tot;
}
void dfs(int root, int fa, long long dis) {
if (dis > maxv) maxv = dis, maxi = root;
for (int x = head[root]; x; x = edges[x].next)
if (edges[x].to != fa) dfs(edges[x].to, root, dis + edges[x].value);
}
map<P, long long> save;
long long getd(int root, int fa, int target) {
if (root == target) return 0;
if(save.count(make_pair(root, target)))return save[make_pair(root, target)];
for (int x = head[root]; x; x = edges[x].next) {
if (edges[x].to == fa) continue;
long long r = getd(edges[x].to, root, target);
if (r != -1) {
save[make_pair(edges[x].to, target)] = r;
return r + edges[x].value;
}
}
return -1;
}
int main() {
scanf("%d%d", &n, &m);
while (m--) {
scanf("%d%d%d", &ix, &iy, &iv);
add(ix, iy, iv);
add(iy, ix, iv);
}
dfs(1, 0, 0);
int d1 = maxi;
maxv = 0, maxi = 0;
dfs(d1, 0, 0);
int d2 = maxi;
long long cmaxv = 0;
int cmaxi = 0;
for (int i = 1; i <= n; i++)
if (i != d1 && i != d2)
cmaxv = max(cmaxv, min(getd(i, 0, d1), getd(i, 0, d2)));
printf("%lld\n", cmaxv + maxv);
return 0;
}
P4408 逃学的小孩 题解的更多相关文章
- 洛谷 P4408 逃学的小孩 解题报告
P4408 [NOI2003]逃学的小孩 题目描述 Chris家的电话铃响起了,里面传出了Chris的老师焦急的声音:"喂,是Chris的家长吗?你们的孩子又没来上课,不想参加考试了吗?&q ...
- 洛谷 P4408 [NOI2003] 逃学的小孩 题解
Analysis 题意虽然说先去谁家再去谁家,但是我们不需要管这个,因为AA.BB.CC三个点我们可以任意互相交换它们所代表的对象,所以题目要求的就是在一棵树上找到3个点AA.BB.CC令AB+BCA ...
- BZOJ1509 & 洛谷4408:[NOI2003]逃学的小孩——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=1509 https://www.luogu.org/problemnew/show/P4408 sb ...
- 洛谷P4408 逃学的小孩
题目 求树的直径,因为任意两个居住点之间有且只有一条通路,所以这是一棵树. 根据题意父母先从C去A,再去B,或者反过来. 我们一定是要让A到B最大,也要让C到A和B的最小值最大. AB最大一定就是直径 ...
- [NOI2003]逃学的小孩 题解
前言 >原题传送门(洛谷)< 看了一下洛谷题面,这道NOI的题竟然是蓝的(恶评?),做了一下好像确实是蓝的... 解法 思路非常简单,找道树的直径,然后答案是直径长度加上最大的min(di ...
- 【BZOJ1509】[NOI2003]逃学的小孩 直径
[BZOJ1509][NOI2003]逃学的小孩 Description Input 第一行是两个整数N(3 N 200000)和M,分别表示居住点总数和街道总数.以下M行,每行给出一条街道的 ...
- BZOJ 1509[NOI 2003]逃学的小孩 树形dp
1509: [NOI2003]逃学的小孩 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 995 Solved: 505[Submit][Status][ ...
- [NOI2003]逃学的小孩(树的直径)
[NOI2003]逃学的小孩 题目描述 Chris家的电话铃响起了,里面传出了Chris的老师焦急的声音:"喂,是Chris的家长吗?你们的孩子又没来上课,不想参加考试了吗?"一听 ...
- BZOJ 1509: [NOI2003]逃学的小孩( 树形dp )
树形dp求出某个点的最长3条链a,b,c(a>=b>=c), 然后以这个点为交点的最优解一定是a+2b+c.好像还有一种做法是求出树的直径然后乱搞... ----------------- ...
随机推荐
- Python之大数据库hive实战
今天和大家分享的是Python如何连接hive数据库来进行hivesql的查询操作. step1:环境准备 Python版本:3.6.2 Windows版本:Windows10版本的64位 ste ...
- CSS清除浮动&内容居中&文字溢出
学习! 1.CSS清除浮动的方法 (1)添加标签清除浮动: 在浮动元素结尾处,并列的添加标签<div style="clear:both;"></div>. ...
- OAuth + Security - 6 - 自定义授权模式
我们知道OAuth2的官方提供了四种令牌的获取,简化模式,授权码模式,密码模式,客户端模式.其中密码模式中仅仅支持我们通过用户名和密码的方式获取令牌,那么我们如何去实现一个我们自己的令牌获取的模式呢? ...
- Java规则引擎 Easy Rules
1. Easy Rules 概述 Easy Rules是一个Java规则引擎,灵感来自一篇名为<Should I use a Rules Engine?>的文章 规则引擎就是提供一种可选 ...
- pip install 执行过程中遇到的各种问题
一.pip install 安装指定版本的包 要用 pip 安装指定版本的 Python 包,只需通过 == 操作符 指定. pip install robotframework == 2.8.7 将 ...
- FWT,FST入门
0.目录 目录 0.目录 1.什么是 FWT 2. FWT 怎么做 2.1. 或卷积 2.2.与卷积 2.3.异或卷积 2.4.例题 3. FST 3.1. FST 怎么做 3.2.例题 1.什么是 ...
- Dorado开发——树形下拉框
最近在学习Dorado开发的过程中,遇到了一个问题,Dorado的树形下拉框选择:Dorado默认情况下父节点和子节点都是可选的,而我要实现的是父节点不可选. 解决办法:在下拉框中,判断父子节点,点击 ...
- windbg分析一次大查询导致的内存暴涨
项目上反馈了一个问题,就是在生产环境上,用户正常使用的过程中,出现了服务器内存突然暴涨,客户有点慌,想找下原因. 讲道理,内存如果是缓慢上涨一直不释放的话,应该是存在内存泄漏的,这种排查起来比较困难, ...
- ZooKeeper使用入门
ZooKeeper简介 ZooKeeper是一个分布式的,开源的分布式应用程序协调服务,是Hadoop的子项目之一.它是一个为分布式应用提供一致性服务的软件,提供的功能包括:配置维护.域名服务.分布式 ...
- git 提交流程
Git提交流程: 1. Menu remote > (拉取)fetch 2. 重新扫描(rescan) 3. 缓存改动(stage change) 4. 写注释后提交(commit) 5. Me ...