传送门

Luogu

解题思路

一眼先二分(上界树的直径,下界最小边权),然后再考虑 \(\text{DP}\)。

对于当前节点 \(u\),在它的所有儿子中分别返回一条匹配不完的长度最大的路径 \(Max\)。

若该路径长大于二分值,直接修一条,不然丢进 \(\text{multiset}\) 里面。

对于 \(\text{multiset}\) 里的元素每次贪心的找出尽可能大的一条与最小的匹配,若找不到则用来更新 \(Max\)

\(check\) 函数里面返回 \(ans\ge m\),最后输出答案即可。

细节注意事项

  • \(\text{multiset}\) 的使用要熟练

参考代码

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <cctype>
#include <cmath>
#include <ctime>
#include <set>
#define rg register
using namespace std;
template < typename T > inline void read(T& s) {
s = 0; int f = 0; char c = getchar();
while (!isdigit(c)) f |= (c == '-'), c = getchar();
while (isdigit(c)) s = s * 10 + (c ^ 48), c = getchar();
s = f ? -s : s;
} const int _ = 50010;
const int __ = 100010; int tot, head[_], nxt[__], ver[__], w[__];
inline void Add_edge(int u, int v, int d)
{ nxt[++tot] = head[u], head[u] = tot, ver[tot] = v, w[tot] = d; } int n, m, ans;
multiset < int > S[_];
multiset < int > ::iterator it; inline int dfs(int u, int f, int mid) {
S[u].clear();
for (rg int i = head[u]; i; i = nxt[i]) {
int v = ver[i]; if (v == f) continue;
int res = dfs(v, u, mid) + w[i];
if (res >= mid) ++ans; else S[u].insert(res);
}
int _max = 0;
while (!S[u].empty()) {
if (S[u].size() == 1)
return _max = max(_max, *S[u].begin());
it = S[u].lower_bound(mid - *S[u].begin());
if (it == S[u].begin() && S[u].count(*it) == 1) ++it;
if (it == S[u].end()) {
_max = max(_max, *S[u].begin());
S[u].erase(S[u].find(*S[u].begin()));
} else {
++ans;
S[u].erase(S[u].find(*it));
S[u].erase(S[u].find(*S[u].begin()));
}
}
return _max;
} inline bool check(int mid)
{ ans = 0, dfs(1, 0, mid); return ans >= m; } int _min = 100000, _max, id; inline void dfs_d(int u, int f, int sum) {
if (sum > _max) _max = sum, id = u;
for (rg int i = head[u]; i; i = nxt[i]) {
int v = ver[i]; if (v == f) continue;
_min = min(_min, w[i]), dfs_d(v, u, sum + w[i]);
}
} inline void get_d() { dfs_d(1, 0, 0), _max = 0, dfs_d(id, 0, 0); } int main() {
#ifndef ONLINE_JUDGE
freopen("in.in", "r", stdin);
#endif
read(n), read(m);
for (rg int u, v, d, i = 1; i < n; ++i)
read(u), read(v), read(d), Add_edge(u, v, d), Add_edge(v, u, d); get_d(); int l = _min, r = _max;
while (l < r) {
int mid = (l + r + 1) >> 1;
if (check(mid)) l = mid;
else r = mid - 1;
} printf("%d\n", l);
return 0;
}

完结撒花 \(qwq\)

最新文章

  1. 关于计算机改名无法连接TFS的问题
  2. 【Linux】解决Wesnoth中文乱码问题
  3. hive学习3(hive基本操作)
  4. Python:字符串
  5. 【Sqlserver】企业管理器打不开
  6. linux远程执行命令
  7. light oj 1294 - Positive Negative Sign
  8. atitit.提升研发效率的利器---重型框架与类库的差别与设计原则
  9. express学习
  10. ABP官方文档翻译 6.3 本地化
  11. C#的排序Sort和OrderBy扩展方法
  12. Django自定义分页
  13. 04-Python入门学习-流程控制
  14. lua时间戳和日期转换及踩坑
  15. zookeeper安装(集群)
  16. dokuwiki 安装配置
  17. 【转】详解在visual studio中使用git版本系统(图文)
  18. 进程池的同步方法 pool.apply
  19. [django]drf知识点梳理-搜索
  20. 【css】文本超出行数以省略号显示

热门文章

  1. laravel如何A表中包含B表中信息
  2. HTML列表标签
  3. CentOS7下使用C/C++连接MariaDB/MySQL
  4. 19年7月份面试7家公司,整理的java面试题(答案自行百度解决,也是个学习的过程)
  5. 【C语言】输入三个正整数a,b,c,求最大值,要求定义一个计算最大值的函数max(a,b),返回a,b的值
  6. 远程操作Linux主机
  7. Python 多任务(进程) day1(2)
  8. 前端——语言——Core JS——《The good part》读书笔记——第三章节(Object)
  9. Java - Test - TestNG: testng.xml 元素 class
  10. jmeter巧用自增长型变量