题目戳我

\(\text{Solution:}\)

根据题目信息简化题意,是让你在树上找出\(m\)条路径使得路径长度最小值最大。

看到题第一感先二分一个答案,问题转化为如何选择一些路径使得它们最小值都大于当前二分的答案。

观察一个节点的子树,必然会带着若干链。它们有些合法,有些需要拼凑成一条合法路径。

于是,我们令\(val[x]\)表示这条路径以\(x\)结尾的长度。将\(x\)所连接的所有路径提取出来。

贪心地,我们显然要使得\(x\)所连的这一条路径最大。所以在给其他路径拼凑的时候要用尽量小的链来凑出合法路径。

于是,我们对所有提取出的路径排序并二分,找到和它可以匹配的最优路径,并标记。

对于二分到的点,我们可以向后枚举直到一个未被标记的点。

最后复杂度是\(O(n\log^2 n).\)注意\(vis\)数组用当前点做编号可以免去清空。

#include<bits/stdc++.h>
using namespace std;
const int MAXN=1e5+10;
const int inf=(1<<30);
int n,m,tot=1,head[MAXN],rt,Siz[50001],mson[50001],M;
int val[50001],up,vis[100010],q[100010];
struct E{int nxt,to,dis;}e[MAXN];
inline void add(int x,int y,int w){
e[++tot].to=y;e[tot].nxt=head[x];
e[tot].dis=w;head[x]=tot;
}
int Gr(int x,int fa){
int sum1=0,sum2=0;
Siz[x]=1;mson[x]=0;
for(int i=head[x];i;i=e[i].nxt){
int j=e[i].to;
if(j==fa)continue;
sum2=max(sum2,Gr(j,x)+e[i].dis);Siz[x]+=Siz[j];
if(Siz[j]>mson[x])mson[x]=Siz[j];
if(sum1<sum2)swap(sum1,sum2);
}
if(n-Siz[x]>mson[x])mson[x]=n-Siz[x];
if(M>mson[x])M=mson[x],rt=x;
up=max(up,sum1+sum2);
return sum1;
}
inline bool checkedge(int v1,int v2,int ck){return (v1+v2)>=ck;}
int dfs(int x,int fa,int v){
int cnt=0,ans=0;
if(ans>=m)return ans;
for(int i=head[x];i;i=e[i].nxt){
int j=e[i].to;
if(j==fa)continue;
ans+=dfs(j,x,v);
}
for(int i=head[x];i;i=e[i].nxt){
int j=e[i].to;
if(j==fa)continue;
if(val[j]+e[i].dis>=v){
ans++;
continue;
}
q[++cnt]=e[i].dis+val[j];
}
sort(q+1,q+cnt+1);
for(int i=1;i<=cnt;++i){
if(vis[i]==x)continue;
int L=i+1,R=cnt,AA=0;
while(L<=R){
int mid=L+R>>1;
if(checkedge(q[i],q[mid],v))AA=mid,R=mid-1;
else L=mid+1;
}
while(vis[AA]==x&&AA<=cnt)AA++;
if(AA>cnt)continue;
if(vis[AA]==x)continue;
if(AA&&checkedge(q[i],q[AA],v)){
vis[AA]=x;vis[i]=x;
ans++;continue;
}
}
for(int i=cnt;i>=1;--i){
if(vis[i]==x)continue;
val[x]=q[i];break;
}
return ans;
}
bool check(int x){
for(int i=1;i<=n;++i)val[i]=0,vis[i]=vis[i+n]=-1;
int A=dfs(rt,0,x);
return A>=m;
}
int main(){
scanf("%d%d",&n,&m);
int l=1,r;
for(int i=1;i<n;++i){
int x,y,z;
scanf("%d%d%d",&x,&y,&z);
add(x,y,z);add(y,x,z);
}
M=inf;Gr(1,0);r=up;
int 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;
}

【题解】NOIP2018 赛道修建的更多相关文章

  1. 竞赛题解 - NOIP2018 赛道修建

    \(\mathcal {NOIP2018}\) 赛道修建 - 竞赛题解 额--考试的时候大概猜到正解,但是时间不够了,不敢写,就写了骗分QwQ 现在把坑填好了~ 题目 (Copy from 洛谷) 题 ...

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

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

  3. Luogu5021 [NOIP2018]赛道修建

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

  4. [NOIP2018]赛道修建(二分+multiset)

    考场上打了一个 \(vector\) 解法,因为我当时不会 \(multiset\) 好吧,我来讲一讲今年的 \(tgD1T3\) 首先,这题 \(55\) 分是不难想的 1. \(b_i=a_i+1 ...

  5. 【比赛】NOIP2018 赛道修建

    最小值最大,二分长度 然后判断赛道大于等于这个长度最多可以有多少条 可以贪心,对于一个点和它的一些儿子,儿子与儿子之间尽量多配(排序后一大一小),剩下的选个最长的留给自己的父亲就好了 具体实现可以用一 ...

  6. [NOIP2018]赛道修建

    嘟嘟嘟 因为一些知道的人所知道的,不知道的人所不知道的原因,我来改写今年的NOIP了. 现在看这题,心中满是疑问:我当时是多么的zz,这种水题为啥没做出来-- 不管了,说正事. 先考虑部分分. 1.\ ...

  7. luogu5021 [NOIp2018]赛道修建 (二分答案+dp(贪心?))

    首先二分一下答案,就变成了找长度>=m的 不相交的路径的个数 考虑到在一个子树中,只有一个点能出这个子树去和别的点搞 所以我这个子树里尽量自我满足是不会有坏处的 而且要在自我满足数最大的条件下, ...

  8. 题解 NOIP2018【赛道修建】—— 洛谷

    这道题有一点点树上dp的意思(大佬轻喷 我刚拿到这道题的时候毫无头绪,只知道这道题要二分答案 为什么是二分答案??? 题目: 目前赛道修建的方案尚未确定.你的任务是设计一 种赛道修建的方案,使得修建的 ...

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

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

随机推荐

  1. TinkPHP框架开发的CRMEB小程序商城v4.0二次开发集成支付宝支付

    前言 大家都知道支付宝支付和微信支付宝都只能局限在自己的平台,微信内支付宝支付是根本就不能使用,即使是公众号支付也需要跳转到外部浏览器才可以唤起支付宝支付,并且QQ浏览器唤起支付宝支付还是问题很多,所 ...

  2. Docker 网络模式详解及容器间网络通信

    当项目大规模使用 Docker 时,容器通信的问题也就产生了.要解决容器通信问题,必须先了解很多关于网络的知识.Docker 作为目前最火的轻量级容器技术,有很多令人称道的功能,如 Docker 的镜 ...

  3. Android反解符号表工具

    cd ~/android-ndk-r13b/toolchains/arm-linux-androideabi-4.9/prebuilt/darwin-x86_64/bin ./arm-linux-an ...

  4. 终于开始了,微软的野心将通过全场景开发平台dotnet 5体现得淋漓尽致!

      本文已经同步到微信公众号「极客起源」 . 现在都在谈论全场景开发,也就是用一套开发工具,可以开发包括但不限于桌面.移动.IOT.游戏.Web等平台的应用.这样对于开发人员是非常爽的.本文将介绍微软 ...

  5. jQuery提供的Ajax方法

    jQuery提供了4个ajax方法:$.get()  $.post()  $.ajax()  $.getJSON() 1.$.get() $.get(var1,var2,var3,var4): 参数1 ...

  6. 推荐掌握Linux shell中这7种运算命令

    #常见的算术运算符号 .+.-:加减 .*./.%:乘.除.取余 .**:幂运算 .++.--:增加记减少 .!.&&.||:取反,并且,或 .<,<=,>,=> ...

  7. selenium常用webdriver api汇总

    1.driver.current_url:用于获得当前页面的URL 2.driver.title:用于获取当前页面的标题 3.driver.page_source:用于获取页面html源代码 4.dr ...

  8. SpringBoot2.0 配置多数据源

    一.简述 配置多数据源意思就是在一个项目中使用多个数据库,在项目使用中可以不用手动切换数据库来实现不同数据库的数据获取和更新. 源码地址: https://github.com/hanguilin/b ...

  9. softmax交叉熵损失函数求导

    来源:https://www.jianshu.com/p/c02a1fbffad6 简单易懂的softmax交叉熵损失函数求导 来写一个softmax求导的推导过程,不仅可以给自己理清思路,还可以造福 ...

  10. 第2课 - 初识makefile的结构

    第2课 - 初识makefile的结构 1. makefile 的意义 (1)makefile 用于定义源文件之间的依赖关系 (在阅读开源软件源码时,可通过Makefile掌握源码中各个文件之间的关系 ...