[NOIP2015 提高组] 运输计划题解
题目链接:P2680 [NOIP2015 提高组] 运输计划 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)
看了好长时间题解才终于懂的,有关lca和二分答案的题解解释的不详细,一时半会理解不过来,于是自己写一篇解释尽管解释主要在代码中,希望能对迷茫的小伙伴有帮助
解析(主要为二分答案的解析,lca只是求距离和找覆盖边时用得到,这里不多说):
由于m个运输计划是同时出发,所以所需要的时间取决于花费最长的时间,因为一个任务在y分钟内完成,那么另一个任务x(x<=y)也一定完成。
问题就转化成了求最长边最短,二分答案。
接下来,就是怎么二分了
想让时间大于答案的任务的时间减小,题目里说了,可以设一条边为虫洞,我们就找这个任务里哪条(些)边可以设为虫洞,这条(些)边就是被覆盖的边,我们将它们记录一下,所以我们只要找这些超时的任务的公共覆盖边就好了,即这条(些)边被记录的次数等于超时的任务数
代码有详细注释,不懂可以去看一下代码
#include<bits/stdc++.h>
#define ll long long
#define rll register long long
using namespace std;
const ll N=3e5+5;
ll n,m,cnt;
ll h[N],lg[N],sum[N],deep[N],st[N][20],vis[N],up[N];
struct edge//边
{
ll v,w,nxt;
} e[N<<1];
struct rw//任务
{
ll u,v,lca,dis;
bool operator <(const rw x)//重载运算符
{
return dis<x.dis;
}
} g[N];
inline ll read()
{
ll x=0;
bool flag=false;
char ch=getchar();
while(ch<'0'||ch>'9')
{
if(ch=='-') flag=true;
ch=getchar();
}
while(ch>='0'&&ch<='9')
{
x=(x<<3)+(x<<1)+(ch^48);
ch=getchar();
}
return flag?~x+1:x;
}
void add(ll u,ll v,ll w)
{
e[++cnt].v=v;
e[cnt].w=w;
e[cnt].nxt=h[u];
h[u]=cnt;
}
void dfs(ll u,ll fat)
{
deep[u]=deep[fat]+1;//深度更新
st[u][0]=fat;//更新st表
for(rll i=1;i<=lg[deep[u]];++i)
{
st[u][i]=st[st[u][i-1]][i-1];//更新st表
}
for(rll i=h[u];i;i=e[i].nxt)//遍历
{
ll v=e[i].v,w=e[i].w;
if(v!=fat)//如果不是父节点
{
sum[v]=(sum[u]+w);
//到根节点的距离就是父节点到根节点的距离加上这条边的边权
up[v]=w;//处理到父节点的距离
dfs(v,u);//继续搜索
}
}
}
ll LCA(ll x,ll y)//正常求LCA
{
if(deep[x]<deep[y])
{
x=x^y,y=x^y,x=x^y;//位运算版本的交换
}
while(deep[x]>deep[y])
{
x=st[x][lg[deep[x]-deep[y]]-1];//让两点在同一深度
}
if(x==y) return x;
for(rll i=lg[deep[x]]-1;i>=0;--i)//找交汇之前最后到达的点
{
if(st[x][i]!=st[y][i])//如果不相交,向上跳
{
x=st[x][i];
y=st[y][i];
}
}
return st[x][0];//返回lca
}
bool check(ll len)
{
if(g[m].dis<=len) return true;
//如果路程最长的都比不过当前二分的答案,那么这个答案是可行的,直接返回
for(rll i=1;i<=n;++i) vis[i]=0;//每次操作前,数组清0
ll cont=0,maxn=0;
//cont 记录比当前答案大的边,maxn记录被覆盖次数最多的边被覆盖的次数
//(我知道maxn这里我解释的很trouble,不好理解就把他看成每条边被覆盖的次数的最大值)
//一定要时刻记住maxn的含义!!!
for(rll i=m;i>=1;--i)//遍历
{
if(g[i].dis<=len) break;
//如果当前任务的路程已经小于答案,就没必要继续遍历查找了,退出即可
cont++;//当前任务路程比答案大,计数器+1
ll u=g[i].u,v=g[i].v,lca=g[i].lca,c=g[i].dis-len;
//c 如果这条边大于或等于c,就说明删去它,这个任务的时间就小于或等于答案
//即这条边可以设为虫洞
while(u!=lca)//如果起点不是lca,那就向上找,查询被覆盖的边
{
if(up[u]>=c)//大于c,说明在这个任务中,它可以设为虫洞,也就是上文说的覆盖
{
vis[u]++;
maxn=max(maxn,vis[u]);//更新maxn
}
u=st[u][0];//往上慢慢找,时间限制是1s~2s
}
while(v!=lca)//如果终点不是lca,向上找
{
if(up[v]>=c)//和上面一样
{
vis[v]++;
maxn=max(maxn,vis[v]);//更新maxn
}
v=st[v][0];//慢慢跳(~ ̄▽ ̄)~
}
if(maxn<cont)return false;
//这里maxn只能小于等于cont,因为进行了cont次循环,一条边最多被覆盖cont次
//maxn是边被覆盖的次数的最大值
//如果小于cont,就说明找不到一条边可以被删去后使任务的时间小于等于答案
//所以这个答案不成立
//如果maxn==cont 就说明至少有一条边可以做到删去后使任务的时间小于等于答案
}
return true;
}
int main()
{
n=read(),m=read();
for(rll i=1;i<n;++i)
{
ll x=read(),y=read(),z=read();
add(x,y,z);
add(y,x,z);
}
for(rll i=1;i<=n;++i)
{
lg[i]=lg[i-1]+(1<<lg[i-1]==i);//预处理log,节约时间
}
dfs(1,0);//搜索
for(rll i=1;i<=m;++i)//记录每一个计划的信息
{
ll u=read(),v=read(),lca=LCA(u,v);
g[i].u=u;//起点
g[i].v=v;//终点
g[i].lca=lca;//lca
g[i].dis=sum[u]+sum[v]-(sum[lca]<<1);//这个计划的长度
}
sort(g+1,g+m+1);//将任务从小到大排序
ll l=0,r=3e8,ans;
while(l<r)
//注意!!!这里二分的是最终答案,也就是最小花费时间,清楚概念,否则后边就和我一样容易混
{
ll mid=(l+r)>>1;//二分操作
if(check(mid)) r=mid,ans=mid;
else l=mid+1;
}
printf("%lld\n",ans);
return 0;
}
[NOIP2015 提高组] 运输计划题解的更多相关文章
- NOIP2015 提高组] 运输计划
码农题啊兄弟们. 随便考虑二分一下,然后发现要取一条满足性质的边. 被所有大于\(mid\)的路径都覆盖,取了之后能把他们都弄到小于\(mid\) 那就树上差分再处理一下. 写了\(180h\),老年 ...
- [NOIP2015提高组]运输计划
题目:BZOJ4326.洛谷P2680.Vijos P1983.UOJ#150.codevs4632.codevs5440. 题目大意:有一棵带权树,有一些运输计划,第i个运输计划从ai到bi,耗时为 ...
- 洛谷 P2680 [NOIP2015 提高组] 运输计划
链接:P2680 题意: 在树上把一条边边权变为0使得最长给定路径最短 分析: 最大值最小可以想到二分答案,对于每一个mid,寻找所有大于mid的路径,再寻找是否存在一条边使得删去它后大于mid的路径 ...
- P2680 [NOIP2015 提高组] 运输计划 (树上差分-边差分)
P2680 题目的大意就是走完m条路径所需要的最短时间(边权是时间), 其中我们可以把一条边的权值变成0(也就是题目所说的虫洞). 可以考虑二分答案x,找到一条边,使得所有大于x的路径都经过这条边(差 ...
- 【数据结构】运输计划 NOIP2015提高组D2T3
[数据结构]运输计划 NOIP2015提高组D2T3 >>>>题目 [题目描述] 公元 2044 年,人类进入了宇宙纪元.L 国有 n 个星球,还有 n−1 条双向航道,每条航 ...
- 【题解】NOIP2015提高组 复赛
[题解]NOIP2015提高组 复赛 传送门: 神奇的幻方 \([P2615]\) 信息传递 \([P2661]\) 斗地主 \([P2668]\) 跳石头 \([P2678]\) 子串 \([P26 ...
- 运输计划(题解)(Noip2015)
运输计划(题解)(Noip2015) 二分答案+树上差分 树上差分其实不难,只是名字高大尚,可以学一下:Eternal风度的树上差分 本人博客里也总结了一些其他的知识供大家学习:Eternal风度的博 ...
- 洛谷 P2678 & [NOIP2015提高组] 跳石头
题目链接 https://www.luogu.org/problemnew/show/P2678 题目背景 一年一度的“跳石头”比赛又要开始了! 题目描述 这项比赛将在一条笔直的河道中进行,河道中分布 ...
- 【二分查找】 跳石头NOIP2015提高组 D2T1
[二分查找]跳石头NOIP2015提高组 D2T1 >>>>题目 [题目描述] 一年一度的“跳石头”比赛又要开始了! 这项比赛将在一条笔直的河道中进行,河道中分布着一些巨大岩石 ...
随机推荐
- 逆向进阶,利用 AST 技术还原 JavaScript 混淆代码
什么是 AST AST(Abstract Syntax Tree),中文抽象语法树,简称语法树(Syntax Tree),是源代码的抽象语法结构的树状表现形式,树上的每个节点都表示源代码中的一种结构. ...
- mongodb 复杂查询
记录一下工作中用到的 mongodb 复杂查询 aggregate 筛选 === 等于 { $match: { name: "bob" } } !== 不等于 { $match: ...
- c 语言彩票选号
最近刚学了c语言,就做了个彩票选号程序练手玩玩,做的不好请见谅 1.分为前区(1-35)和后区(1-12)号码 2.先循环随机前区号在循环后区号 3.生成随机时数判断是否有重复值,和之前5期是否出现过 ...
- 个人冲刺(二)——体温上报app(一阶段)
任务:完成了WenData类的编写,同时完成了SecondActivity.java SecondActivity.java package com.example.helloworld; impor ...
- [codeforces] 暑期训练之打卡题(一)
每个标题都做了题目原网址的超链接 Day1<Vanya and Lanterns> 题意: 一条长度为 l 的街道,在这条街道上放置了n个相同的灯,街道一端位置记为0,每个灯的位置在ai处 ...
- 基于surging网络组件多协议适配的平台化发展
前言 Surging 发展已经有快6年的时间,经过这些年的发展,功能框架也趋于成熟,但是针对于商业化需求还需要不断的打磨,前段时间客户找到我想升级成平台化,针对他的需求我 ...
- Java实现飞机大战游戏
飞机大战详细文档 文末有源代码,以及本游戏使用的所有素材,将plane2文件复制在src文件下可以直接运行. 实现效果: 结构设计 角色设计 飞行对象类 FlyObject 战机类 我的飞机 MyPl ...
- Android 子线程 UI 操作真的不可以?
作者:vivo 互联网大前端团队- Zhang Xichen 一.背景及问题 某 SDK 有 PopupWindow 弹窗及动效,由于业务场景要求,对于 App 而言,SDK 的弹窗弹出时机具有随机性 ...
- sqlserver 插入 更新 删除 语句中的 output子句
官方文档镇楼: https://docs.microsoft.com/zh-cn/previous-versions/sql/sql-server-2008/ms177564(v=sql.100) 从 ...
- 重载overload 、重写override
观点:重载和重写完全没有关系要联系到一起,唯一的联系就是他们都带有个'重'字,所以鄙人也随大流把他们放在了一起 注意:下面可复制的代码是正确的,错误的只会上传图片,不上传可复制的代码 重载 1.在同一 ...