#\(\mathcal{\color{red}{Description}}\)

\(Link\)

在一棵带有边权的树上,可以选择使一条边权为零。然后对于所有\(M\)条链,使其链长最大值最小。

#\(\mathcal{\color{red}{Solution}}\)

……还是老方法,先分析贪心是否可行\(->\)不可行。再分析状态如何定义\(->\)不可定义。最后看比赛难度\(->NOIP\)。得出结论:不是线性规划而是二分答案

嗯,好的,在这之后我们就可以愉悦地二分答案——二分链长最大值最小值,那我们接下来就要考虑如何\(check\)了。我们思考二分出的最长链的长度之后,我们要找一条大于二分出来的长度的链的公共边,让它变成零——设窝萌二分出的链长是\(k\),大于\(k\)的链有\(s\)条,那么我们考虑这\(s\)条可能有\(n(0 \leq n \leq m)\)条公共边,我们选择当\(n\)不为零的时候直接选一条最大的公共边,让最长的链减去这条边的权值,观察其是否\(\leq\)我们二分得到的\(k\),如果是就\(return \ \ 1\)调整\(r\)了;那如果是\(n=0\),这种方案一定会无从下手,因为作为最终答案来讲,若最短时间为\(T\),那么所有原来权值大于\(T\)的边一定有起码一条公共边,所以调整\(l\),直至二分结束即可。

嗯……一二分答案就蒙○……

哦对,链长我们是可以预处理出来的\(qwq\)

#include <cmath>
#include <cstdio>
#include <iostream>
#define MAXN 600010 using namespace std ;
struct edge{
int to, next, v ;
}e[MAXN] ; int cnt, head[MAXN] ;
int LcA[MAXN], fa[MAXN][32], dep[MAXN], Len[MAXN] ;
int l = 1, r, mid, i, j, A, B, C, N, M, Up, Max, pre, num ;
int dif[MAXN], S[MAXN], T[MAXN], dis[MAXN], edges[MAXN], res, ans ; inline int qr(){
int k = 0 ; char c = getchar() ;
while(!isdigit(c)) c = getchar() ;
while(isdigit(c)) k = (k << 1) + (k << 3) + c - 48, c = getchar() ;
return k ;
}
inline void add(int u, int v, int w){
e[++ cnt].to = v, e[cnt].v = w ;
e[cnt].next = head[u], head[u] = cnt ;
e[++ cnt].to = u, e[cnt].v = w ;
e[cnt].next = head[v], head[v] = cnt ;
}
void _build(int deep, int now, int f){
fa[now][0] = f ; dep[now] = deep ;
for(int k = head[now]; k ;k = e[k].next){
if(e[k].to == f) continue ;
edges[e[k].to] = e[k].v ;
Len[e[k].to] = Len[now] + edges[e[k].to] ;
_build(deep + 1, e[k].to, now) ;
}
}
inline void _get(int now){
for(int k = head[now]; k ; k = e[k].next){
if(e[k].to == fa[now][0]) continue ;
_get(e[k].to) ;
dif[now] += dif[e[k].to] ;
}
if(dif[now] == num && edges[now] > res) res = edges[now] ;
}
inline void init(){
Up = log(N) / log(2) + 1 ;
for(i = 1; i <= Up; i ++)
for(j = 1; j <= N; j ++)
fa[j][i] = fa[fa[j][i - 1]][i - 1] ;
}
inline int LCA(int u, int v){
if(dep[u] < dep[v]) swap(u, v) ;
pre = dep[u] - dep[v] ;
for(j = 0; j <= Up; ++ j) if((1 << j) & pre) u = fa[u][j] ;
if(u == v) return u ;
for(j = Up; j >= 0; -- j) if(fa[u][j] != fa[v][j]) u = fa[u][j], v = fa[v][j] ;
return fa[v][0] ;
}
inline bool check(int x){
fill(dif + 1, dif + N + 1, 0) ; num = 0 ; res = 0 ;
for(i = 1; i <= M; ++ i)
if(dis[i] > x)
num ++, dif[S[i]] ++, dif[T[i]] ++, dif[LcA[i]] -= 2 ;
_get(1) ;
if(Max - res > x) return 0 ;
return 1 ;
} int main(){
N = qr(), M = qr() ;
for(i = 1; i < N; ++ i)
A = qr(), B = qr(), C = qr(), add(A, B, C);
_build(1, 1, 0) ; init() ;
for(i = 1; i <= M; ++ i){
S[i] = qr(), T[i] = qr(), LcA[i] = LCA(S[i], T[i]) ;
dis[i] = Len[S[i]] + Len[T[i]] - (Len[LcA[i]] << 1) ;
Max = max(dis[i], Max) ;
}
r = Max ;
while(l <= r){
mid = (l + r) >> 1 ;
if(check(mid)) r = mid - 1, ans = mid;
else l = mid + 1 ;
}
cout << ans ;
}

本题心得:

\(1\)、有时候单纯暴力的二分不是很好做……但是我们如果换种思路思考二分的话,他的显然性就很显然。

\(2\)、嗯,我还是太弱了,未来的路好长啊……

随机推荐

  1. Fastify 系列教程一 (路由和日志)

    Fastify 系列教程: Fastify 系列教程一 (路由和日志) Fastify 系列教程二 (中间件.钩子函数和装饰器) Fastify 系列教程三 (验证.序列化和生命周期) Fastify ...

  2. CSS 画一个八卦

    效果图: 实现原理: 设置高度为宽度的2倍的一个框,利用 border 补全另一半的宽度,设置圆角 用两个 div 设置不同的颜色,定位到圆的上下指定位置. 最后只剩下里面的小圆圈了.设个宽高,圆角即 ...

  3. ArcEngine对Blob字段赋值的方法

    今天在测试数据入库程序,发现对某个图层操作之后,调用StopOperation,会出现“尝试写入或读取受保护的内存”错误. 经过测试,最终发现是因为该图层包含有Blob字段,而代码没有专门对Blob字 ...

  4. 网络I/O模型--06异步I/O

    异步I/O (又称为 AIO )则是采用“订阅一通知”工作模式 : 即应用程序向操作系统注册I/O监听,然后继续做自己的事情.当操作系统发生I/O事件,并且准备好数据后 , 再主动通知应用程序,触发相 ...

  5. 用flutter写一个精美的登录页面

    先看效果图: 源代码已上传到github 我们先看一下页面 , 首先这个页面,我们并没有用到AppBar,当然也就没有自带返回功能.然后下面有个Login的文字以及一条横线. 屏幕中上方是填写帐号以及 ...

  6. 【Python】Java程序员学习Python(二)— 开发环境搭建

    巧妇难为无米之炊,我最爱的还是鸡蛋羹,因为我和鸡蛋羹有段不能说的秘密. 不管学啥,都要有环境,对于程序员来说搭建个开发环境应该不是什么难题.按顺序一步步来就可以,我也只是记录我的安装过程,你也可以滴. ...

  7. webpack必知必会

    细节 url-loader和file-loader是什么关系? file-loader用于将文件路径打包为另一个url,url-loader封装了file-loader.使用url-loader时,只 ...

  8. 如何修改ionic中android程序的包名

    默认ionic新建工程的时候指定的Android版本包名是:com.ionicframework.starter:这样固定死包名的话会导致一个问题,多个ionic工程无法正常安装到手机当中,后面安装的 ...

  9. Oracle EBS 获取用户挂的职责 请求 请求的类别(RTF还是什么的)

    select fu.user_ID, fu.user_name, fu.start_date, fu.END_DATE, fu.description, fe.last_name, fr.RESPON ...

  10. Vue2学习笔记:事件对象、事件冒泡、默认行为

    1.事情对象 <!DOCTYPE html> <html> <head> <title></title> <meta charset= ...