P5021 赛道修建[贪心+二分]
题目描述
C 城将要举办一系列的赛车比赛。在比赛前,需要在城内修建 mm 条赛道。
C 城一共有 nn 个路口,这些路口编号为 1,2,…,n1,2,…,n,有 n-1n−1 条适合于修建赛道的双向通行的道路,每条道路连接着两个路口。其中,第 ii 条道路连接的两个路口编号为 a_ia**i 和 b_ib**i,该道路的长度为 l_il**i。借助这 n-1n−1 条道路,从任何一个路口出发都能到达其他所有的路口。
一条赛道是一组互不相同的道路 e_1,e_2,…,e_ke1,e2,…,e**k,满足可以从某个路口出发,依次经过 道路 e_1,e_2,…,e_ke1,e2,…,e**k(每条道路经过一次,不允许调头)到达另一个路口。一条赛道的长度等于经过的各道路的长度之和。为保证安全,要求每条道路至多被一条赛道经过。
目前赛道修建的方案尚未确定。你的任务是设计一种赛道修建的方案,使得修建的 mm 条赛道中长度最小的赛道长度最大(即 mm 条赛道中最短赛道的长度尽可能大)
解析
NOIp的题真的是越来越难啊。。。
首先容易看出是个二分。但是看出来还不够,这题检验就比较恶心,我没想出来(太菜了)。
一开始的想法是计算出一个根下所有点对的路径长,然后贪心地每次把满足二分答案的路径加进去,结果写完之后自己证伪了这个贪心。
如果这题没注意到一个至关重要的点,是很难写下去的:
对于一颗子树,至多存在一条一个端点在这颗子树、另一个端点在另一颗子树的路径。这是由于一颗子树的根节点只有一个父亲,也就是说,子树的根节点到它父亲只存在一条边。
所以我们只要想个办法把某个二分答案下一颗子树内部最多能形成的路径统计出来。这个东西肯定要自底向上得出,显然一颗子树的路径条数取决于它所有子树的路径条数总和,再加上一些经过这颗子树根节点的路径(也可能没有),经过根节点的路径又分为两种情况,一种是终点在另一颗子树种,一种是终点就是根节点。
考虑一个类似树形dp的东西,方便自底向上统计。
我们要统计两个东西:路径的长度、路径的数量。
路径的长度可以推出路径的数量,因此我们只在dp(似乎不是dp)数组记一个穿过子树根节点的路径长度即可。
若要组成一条穿过某颗子树根节点的路径,必然是要合并两条穿过它某两个儿子的路径。我们考虑这样一个贪心:一定是要先合并出尽可能短的满足当前二分答案的路径。所以我们肯定要先拿最短的那条穿过根的路径去匹配另一条尽可能短的、使得它们长度之和满足二分答案的路径。
若要组成一条终点是根节点的路径,我们肯定也是选择那些尽可能短的满足二分答案的路径。所以在合并路径之前,我们先看这个子树是否存在这样的路径,存在就尽早把它统计上。
在上述过程完成后,我们还要选出一条路径穿过根节点,跟其它子树相连(或者仅与父亲节点)。还是那个贪心原则,我们拿出没有统计过的最长的一条路径贡献上去,这样就能使得其它子树中的路径用的尽量少。
说实话我还是不太相信这个贪心是正确的,考场上我更有可能写个搜索。
参考代码
#include<cstdio>
#include<iostream>
#include<cmath>
#include<cstring>
#include<ctime>
#include<cstdlib>
#include<algorithm>
#include<queue>
#include<set>
#include<map>
#define ll long long
#define N 500010
using namespace std;
inline int read()
{
int f=1,x=0;char c=getchar();
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
return x*f;
}
struct rec{
int next,ver,edge;
}g[N<<1];
int head[N],tot,n,m,res,q[N],tag[N],dp[N];
inline void add(int x,int y,int val)
{
g[++tot].ver=y,g[tot].edge=val;
g[tot].next=head[x],head[x]=tot;
}
inline void check(int x,int fa,int lim)
{
for(int i=head[x];i;i=g[i].next){
int y=g[i].ver;
if(y==fa) continue;
check(y,x,lim);
}
int tail=0;dp[x]=0;
for(int i=head[x];i;i=g[i].next){
int y=g[i].ver,z=g[i].edge;
if(y==fa) continue;
q[++tail]=dp[y]+z;
}
sort(q+1,q+tail+1);
for(int i=tail;i>=1&&q[i]>=lim;--i)
res--,tail--;
for(int i=1;i<=tail;++i){
if(tag[i]!=x){
int l=i+1,r=tail,pos=tail+1;
while(l<=r){
int mid=(l+r)>>1;
if(q[i]+q[mid]>=lim) r=mid-1,pos=mid;
else l=mid+1;
}
while(tag[pos]==x&&pos<=tail) pos++;
if(pos<=tail) tag[pos]=tag[i]=x,res--;
}
}
for(int i=tail;i>=1;--i)
if(tag[i]!=x){dp[x]=q[i];break;}
}
int main()
{
ll maxn=0;
n=read(),res=m=read();
for(int i=1;i<n;++i){
int u=read(),v=read(),l=read();
add(u,v,l),add(v,u,l);maxn+=l;
}
ll l=1,r=maxn;
while(l<=r){
res=m;
ll mid=(l+r)>>1;
memset(tag,0,sizeof(tag));
check(1,0,mid);
if(res>0) r=mid-1;
else l=mid+1;
}
cout<<l-1<<endl;
return 0;
}
P5021 赛道修建[贪心+二分]的更多相关文章
- 二分答案 + multiset || NOIP 2018 D1 T3 || Luogu P5021 赛道修建
题面:P5021 赛道修建 题解:二分答案,用Dfs进行判断,multiset维护. Dfs(x,fa,Lim)用来计算以x为根的子树中有多少符合条件的路径,并返回剩余未使用的最长路径长. 贪心思想很 ...
- 【题解】 P5021赛道修建
[题解]P5021 赛道修建 二分加贪心,轻松拿省一(我没有QAQ) 题干有提示: 输出格式: 输出共一行,包含一个整数,表示长度最小的赛道长度的最大值. 注意到没,最小的最大值,还要多明显? 那么我 ...
- 洛谷P5021 赛道修建 NOIp2018 贪心+二分答案
正解:贪心+LCA+二分答案 解题报告: 想先港下部分分qwq因为我部分分只拿到了10ptsQAQ(时间不够不是理由,其实还是太弱,所以要想很久,所以才时间不够QAQ m=1 找直径长度,完 一条链 ...
- [NOIP2018]赛道修建(二分+multiset)
考场上打了一个 \(vector\) 解法,因为我当时不会 \(multiset\) 好吧,我来讲一讲今年的 \(tgD1T3\) 首先,这题 \(55\) 分是不难想的 1. \(b_i=a_i+1 ...
- P5021 赛道修建 (NOIP2018)
传送门 考场上把暴力都打满了,结果文件输入输出写错了.... 当时时间很充裕,如果认真想想正解是可以想出来的.. 问你 长度最小的赛道长度的最大值 显然二分答案 考虑如何判断是否可行 显然对于一个节点 ...
- 洛谷P5021 赛道修建
题目 首先考虑二分,然后发现最小长度越大的话,赛道就越少.所以可以用最终的赛道个数来判断长度是否合理.问题转化为给定一个长度,问最多有多少条互不重叠路径比这个给定长度大. 考虑贪心,毕竟贪心也是二分c ...
- [NOIp2018] luogu P5021 赛道修建
我同学的歌 题目描述 你有一棵树,每条边都有权值 did_idi.现在要修建 mmm 条赛道,一条赛道是一条连贯的链,且一条边至多出现在一条赛道里.一条赛道的长被定义为,组成这条赛道的边的权值之和. ...
- P5021 赛道修建 题解
原题链接 简要题意: 在一棵树上求 \(m\) 条不相交的路径的最小值的最大值. 本题部分分很多,而且本人也交了 \(27\) 次,所以一定要仔细讲部分分! 算法一 对于 \(b_i = a_i + ...
- noip 2018 day1 T3 赛道修建 贪心_树上问题_multiset
Code: // luogu-judger-enable-o2 #include<bits/stdc++.h> using namespace std; #define maxn 5000 ...
随机推荐
- PHP处理SOAP
1.获取functions try { $client = new SoapClient("http://www.fangbei.org/services/inquiryTracingAnd ...
- 在导入pytorch时libmkl_intel_lp64.so找不到
安装或者更新完pytorch后,运行不了,显示错误: (base) xu@xusu:~$ python Python (default, Dec , ::) [GCC ] :: Anaconda, I ...
- python数据分析4之自动采集数据
1 数据采集的重要性 数据采集是数据挖掘的基础,没有数据,挖掘也没有意义.很多时候,我们拥有多少数据源,多少数据量,以及数据质量如何,将决定我们挖掘产出的成果会怎样 2 四类采集方式 3 如何使用开放 ...
- 基于ELK 7.50搭建elastalert 监控报警和权限控制
ELK+监控报警全步骤 需求: 公司要求对出在windows服务器上的日志进行日志分析并根据关键字进行报警,并配置kibana权限控制.下面为详细步骤 环境: centos 7.6 elk版本7.50 ...
- 9. Scala隐式转换和隐式值
9.1 隐式转换 9.1.1 提出问题 先看一个案例演示,引出隐式转换的实际需要=>指定某些数据类型的相互转化 object boke_demo01 { def main(args: Array ...
- Java学习:数据库连接池技术
本节内容 数据库连接池 Spring JDBC : JDBC Template 数据库连接池 1.概念:其实就是一个容器(集合),存放数据库连接的容器 当系统初始化好后,容器中会申请一些连接对象,当用 ...
- 思维导图xmind的文档保存问题
如果文件名相同,可能最新的文档覆盖以前的.当前活动文档只能有一个,如果有多个,保存后,其他活动文档也被更新了. 新建一个空白doc文档,仅仅是文件名,作为附件导入到xmind中,在xmind中保存后, ...
- MVC学习笔记(六)---遇到的小问题汇总
一.MVC中Controller中返回两个对象的写法如下: , msg = "成功", user = user, userInfo = person }); 二.前台向后台传入带有 ...
- Options of the DB storage of prometheus
参考: // Options of the DB storage. type Options struct { // The timestamp range of head blocks after ...
- Java自学-操作符 关系操作符
Java的关系操作符 关系操作符:比较两个变量之间的关系 > 大于 >= 大于或等于 < 小于 <= 小于或等于 == 是否相等 != 是否不等 示例: public clas ...