题目大意:$NOIP2018\;TG\;D1T3$

题解:题目要求最短的赛道的长度最大,可以想达到二分答案,接着就是一个显然的树形$DP$。

发现对于一个点,它子树中若有两条链接起来比要求的答案大,一定接起来成为一条路径,因为接起来答案一定加一,而传递上去的话不一定。然后对于一条链,一定是找可行的最短的链与它相接,把尽可能长的链传递上去。找最小的可行的链我使用了双向链表(复杂度$O(n)$,右端点总共最多向左移动$n$次,每次最多向右移动$1$次)

卡点:考场上写结束后删除节点后转移到下一个节点时,没有考虑到移动到的节点也被删除的情况(考场上我是真的傻)

C++ Code:

#include <cstdio>
#include <algorithm>
#include <vector>
#include <cctype> namespace R {
int x, ch;
inline int read() {
ch = getchar();
while (isspace(ch)) ch = getchar();
for (x = ch & 15, ch = getchar(); isdigit(ch); ch = getchar()) x = x * 10 + (ch & 15);
return x;
}
}
using R::read; #define maxn 50010
const int TANG_Yx = 20040826;
inline int max(int a, int b) {return a > b ? a : b;}
int head[maxn], cnt;
struct Edge {
int to, nxt, w;
} e[maxn << 1];
inline void add(int a, int b, int c) {
e[++cnt] = (Edge) {b, head[a], c}; head[a] = cnt;
} int n, m, sum, ans;
int k, f[maxn];
inline bool debug(int k) {return true;}
int pre[maxn], nxt[maxn]; std::vector<int> V[maxn];
int dfn[maxn], rnk[maxn], idx, fa[maxn];
int up[maxn]; void dfs1(int u, int fa = 0) {
::fa[u] = fa; rnk[u] = u;
dfn[u] = ++idx;
for (int i = head[u]; i; i = e[i].nxt) {
int v = e[i].to;
if (v != fa) {
up[v] = e[i].w;
dfs1(v, u);
}
}
} inline void work(int u, int fa) {
std::vector<int> &V = ::V[u];
std::sort(V.begin(), V.end());
int sz = V.size();
while (sz && V[sz - 1] >= k) f[u]++, sz--;
int l = 0, r = 1, rem = 0;
if (sz > 0) {
#define End sz
#define Begin (sz + 1)
for (register int i = 0; i < sz; i++) {
pre[i] = i - 1;
nxt[i] = i + 1;
}
nxt[Begin] = 0;
pre[0] = Begin;
nxt[sz - 1] = End;
pre[End] = sz - 1;
while (r < End && l < r) {
while (nxt[r] < End && V[l] + V[r] < k) r = nxt[r];
while (pre[r] > l && pre[r] != Begin && V[l] + V[pre[r]] >= k) r = pre[r];
if (V[l] + V[r] >= k) {
f[u]++;
nxt[pre[l]] = nxt[l];
pre[nxt[l]] = pre[l];
nxt[pre[r]] = nxt[r];
pre[nxt[r]] = pre[r];
if (nxt[pre[l]] != End && pre[nxt[r]] != Begin && pre[nxt[r]] > nxt[pre[l]]) r = pre[nxt[r]];
else r = nxt[r];
l = nxt[pre[l]];
} else l = nxt[l];
if (l == r) r = nxt[r];
}
if (0 <= pre[End] && pre[End] < sz) rem = V[pre[End]];
else rem = 0;
#undef End
#undef Begin
}
if (u != 1) {
::V[fa].push_back(rem + up[u]);
f[fa] += f[u];
}
}
inline bool check(int mid) {
k = mid;
for (register int i = 1; i <= n; i++) V[i].clear(), f[i] = 0;
for (register int I = 1, i = rnk[I]; I <= n; i = rnk[++I]) {
work(i, fa[i]);
}
return f[1] >= m;
} namespace Work1 {
int MAX, ans;
void dfs(int u, int fa = 0, int dep = 0) {
if (dep > MAX) {
MAX = dep;
ans = u;
}
for (int i = head[u]; i; i = e[i].nxt) {
int v = e[i].to;
if (v != fa) {
dfs(v, u, dep + e[i].w);
}
}
}
int main() {
MAX = 0;
dfs(1);
int x = ans;
MAX = 0;
dfs(x);
printf("%d\n", MAX);
return 0;
}
} namespace Work2 {
int pre[maxn];
void dfs(int u, int fa = 0) {
for (int i = head[u]; i; i = e[i].nxt) {
int v = e[i].to;
if (v != fa) {
pre[v] = pre[u] + e[i].w;
dfs(v, u);
}
}
}
bool check(int mid) {
int last = 0, res = 0;
for (int i = 1; i <= n; i++) {
if (pre[i] - last >= mid) {
last = pre[i];
res++;
}
}
return res >= m;
}
int main() {
dfs(1);
int l = 1, r = sum / m, ans = 0;
while (l <= r) {
int mid = l + r >> 1;
if (check(mid)) {
l = mid + 1;
ans = mid;
} else r = mid - 1;
}
printf("%d\n", ans);
return 0;
}
} inline bool cmp(int a, int b) {return dfn[a] > dfn[b];}
bool flag = true;
int main() {
n = read(), m = read();
for (int i = 1, a, b, c; i < n; i++) {
a = read(), b = read(), c = read();
add(a, b, c);
add(b, a, c);
if (a - b != 1 && b - a != 1) flag = false;
sum += c;
}
if (m == 1) {
return Work1::main();
}
if (flag) {
return Work2::main();
}
dfs1(1);
std::sort(rnk + 1, rnk + n + 1, cmp);
int l = 1, r = sum / m;
while (l <= r) {
int mid = l + r >> 1;
if (check(mid)) {
l = mid + 1;
ans = mid;
} else r = mid - 1;
}
printf("%d\n", ans);
return 0;
}

  

[NOIP2018 TG D1T3]赛道修建的更多相关文章

  1. [NOIp2018提高组]赛道修建

    [NOIp2018提高组]赛道修建 题目大意: 给你一棵\(n(n\le5\times10^4)\)个结点的树,从中找出\(m\)个没有公共边的路径,使得第\(m\)长的路径最长.问第\(m\)长的路 ...

  2. noip 2018 D1T3 赛道修建

    noip 2018 D1T3 赛道修建 首先考虑二分答案,这时需要的就是对于一个长度求出能在树中选出来的最多的路径条数.考虑到一条路径是由一条向上的路径与一条向下的路径构成,或者仅仅是向上或向下的路径 ...

  3. noip2018 D1T3 赛道修建

    题目描述 C 城将要举办一系列的赛车比赛.在比赛前,需要在城内修建 mm 条赛道. C 城一共有 nn 个路口,这些路口编号为 1,2,…,n1,2,…,n,有 n-1n−1 条适合于修建赛道的双向通 ...

  4. NOIP2018 旅行 和 赛道修建

    填很久以前的坑. 旅行 给一棵 n 个点的基环树,求字典序最小的DFS序. n ≤ 5000 题解 O(n2) 做法非常显然,枚举断掉环上哪条边然后贪心即可.当然我去年的骚操作只能得88分. O(n ...

  5. NOIP2018 D1T3赛道修建

    题目链接:Click here Solution: 最小值最大,考虑二分一个答案\(k\) 考虑在子树内先匹配,最后传递一个值给自己的父亲(因为每条边只能用一次,所以一颗子树最多传递一个值) 那么我们 ...

  6. [NOIp2018] luogu P5021 赛道修建

    我同学的歌 题目描述 你有一棵树,每条边都有权值 did_idi​.现在要修建 mmm 条赛道,一条赛道是一条连贯的链,且一条边至多出现在一条赛道里.一条赛道的长被定义为,组成这条赛道的边的权值之和. ...

  7. Luogu5021 [NOIP2018]赛道修建

    Luogu5021 [NOIP2018]赛道修建 一棵大小为 \(n\) 的树,边带权.选 \(m\) 条链使得长度和最小的链最大. \(m<n\leq5\times10^4\) 贪心,二分答案 ...

  8. 【LG5021】[NOIP2018]赛道修建

    [LG5021][NOIP2018]赛道修建 题面 洛谷 题解 NOIP之前做过增强版还没做出来\(QAQ\) 一看到题目中的最大值最小,就很容易想到二分答案 重点是考虑如何\(check\) 设\( ...

  9. 【noip2018】【luogu5021】赛道修建

    题目描述 C 城将要举办一系列的赛车比赛.在比赛前,需要在城内修建 mm 条赛道. C 城一共有 nn 个路口,这些路口编号为 1,2,…,n1,2,…,n,有 n-1n−1 条适合于修建赛道的双向通 ...

随机推荐

  1. chrome debugger 调试

    debugger 使用chrome调试时,html页面的js代码中可能不好打断点(因为在jvm中才会有代码) 我一开始是故意在需要断点的后面或前面写个错的alert,通过jvm找到此处,然后在需要的地 ...

  2. 使用百度定位Api获取当前用户登录地址

    最近在做一个商城项目,客户想把网站做成类似于美团的效果,切换地区时,内容也跟随变化.这就要首先解决根据用户id获得地址的问题,最终决定使用百度定位(不适用于搭建反向代理的项目) String url ...

  3. Laravel-admin 当使用Form组件hasMany的时候 进行编辑出现错误 foreach错误的时候解决方案

    我的关联关系原名是 goodImage 修改成 image 之后解决问题 分析得出应该是  laravel会将goodImage 转成 good_image字段  这样就foreach会报错  所以出 ...

  4. admin添加用户时报错:(1452, 'Cannot add or update a child row: a foreign key constraint fails (`mxonline`.`django_admin_l

    在stackoverflow找到答案: DATABASES = { 'default': { ... 'OPTIONS': { "init_command": "SET ...

  5. 文件 I/O字节流

    输入字节流: import java.io.*; public class test_main { public static void main(String[] args) { int n=-1; ...

  6. C语言数据结构(二)

    算法和算法的衡量 一.算法 算法是为了解决某类问题而规定的一个有限长的操作序列.一个算法必须满足以下五个重要特性: 1.有穷性   对于任意一组合法输入值,在执行又穷步骤之后一定能结束,即:算法中的每 ...

  7. KMP算法(查找子序列)

    KMP类似暴力,但是不会和暴力完全一样,回溯到起点. 简单的说  假如   模板链字符串是:        abcabcabcabd        寻找abcabd 在模板链出现的次数,并且输出该次数 ...

  8. MVC中Model 的Key值不建议用非int型

    一次在开发中,key的值用了 byte型,结果插入第一条正常,第二条开始就出错,原因是用byte类型无法实现自动增加1,所以为了方便,建议使用 int型. public virtual byte bk ...

  9. LeetCode:22. Generate Parentheses(Medium)

    1. 原题链接 https://leetcode.com/problems/generate-parentheses/description/ 2. 题目要求 给出一个正整数n,请求出由n对合法的圆括 ...

  10. [网站日志]当Memcached缓存服务挂掉时性能监视器中的表现

    我们用的Memcached缓存服务是阿里云OCS,今天晚上遇到了一次OCS挂掉的情况(计划中的升级),看一下性能监视器中的表现,也许对分析黑色1秒问题有帮助. 应用日志中错误: 2014-06-05 ...