P2680 运输计划

链接

分析:

  二分+树上差分。

  首先可以二分一个答案,那么所有比这个答案大的路径,都需要减去些东西才可以满足这个答案。

  那么减去的这条边一定在所有的路径的交集上。

  那么如果求快速的求出这个交集并判断呢,树剖可以,把所有大于的路径都标记一下,然后判断,复杂度太大了。

  于是用到了树上差分,get新技能。

  在两个端点出标记+1,在lca出标记-2,然后从叶子节点往上更新,对于一条边是交集,那么它标记的次数一定是大于这个答案的个数。然后判断是否满足即可。

  各种优化:1、L,R的范围,2、可以按dis排序,发现没什么用。

代码

 // luogu-judger-enable-o2
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<iostream>
#include<cctype> using namespace std; const int N = ;
struct Edge{
int nxt,to,w;
Edge() {}
Edge(int a,int b,int c) {to = a,w = b,nxt = c;}
}e[];
struct Que{
int u,v,lca,dis;
bool operator < (const Que &A) const {
return dis > A.dis;
}
}q[N];
int head[N],tot;
int deth[N],fa[N],num[N],dis[N],val[N],f[N][],tag[N];
int n,m,dfs_time; inline char nc() {
static char buf[],*p1 = buf,*p2 = buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,,,stdin),p1==p2)?EOF:*p1++;
}
inline int read() {
int x = ,f = ;char ch = nc();
for (; !isdigit(ch); ch=nc()) if(ch=='-') f=-;
for (; isdigit(ch); ch=nc()) x=x*+ch-'';
return x * f;
}
void add_edge(int u,int v,int w) {
e[++tot] = Edge(v,w,head[u]);head[u] = tot;
e[++tot] = Edge(u,w,head[v]);head[v] = tot;
}
void dfs(int u,int fa) {
num[++dfs_time] = u;
deth[u] = deth[fa] + ;
f[u][] = fa;
for (int i=head[u]; i; i=e[i].nxt) {
int v = e[i].to;
if (v == fa) continue;
dis[v] = dis[u] + e[i].w;
val[v] = e[i].w;
dfs(v,u);
}
}
void init() {
for (int j=; j<=; ++j)
for (int i=; i<=n; ++i)
f[i][j] = f[f[i][j-]][j-];
}
int Lca(int u,int v) {
if (deth[u] < deth[v]) swap(u,v);
int d = deth[u] - deth[v];
for (int i=; i>=; --i)
if (d & ( << i)) u = f[u][i];
if ( u == v ) return u;
for (int i=; i>=; --i)
if (f[u][i] != f[v][i])
u = f[u][i],v = f[v][i];
return f[u][];
}
bool check(int x) {
memset(tag,,sizeof(tag));
int limit = ,cnt = ;
for (int i=; i<=m; ++i) {
if (q[i].dis > x) { // 找到所有大于x的
tag[q[i].u] ++; tag[q[i].v] ++; tag[q[i].lca] -= ; // 差分
limit = max(limit,q[i].dis - x); // 最少减去多少
cnt ++;
}
}
if (!cnt) return true; //如果没有大于x的,直接返回true,加上这句快了不少。
for (int i=n; i>=; --i)
tag[f[num[i]][]] += tag[num[i]];
for (int i=; i<=n; ++i)
if (val[i] >= limit && tag[i] == cnt) return true;
return false;
}
int main () {
n = read(),m = read();
int L = ,R = ;
for (int i=; i<n; ++i) {
int u = read(),v = read(),w = read();
add_edge(u,v,w);
L = max(L,w);
}
dfs(,);
init();
for (int i=; i<=m; ++i) {
q[i].u = read(),q[i].v = read();
q[i].lca = Lca(q[i].u,q[i].v);
q[i].dis = dis[q[i].u] + dis[q[i].v] - dis[q[i].lca] * ;
R = max(R, q[i].dis);
}
// sort(q+1,q+m+1);
int ans = R;
while ( L <= R ) {
int mid = ( L + R ) / ;
if (check(mid)) ans = mid,R = mid - ;
else L = mid + ;
}
cout << ans;
return ;
}

P2680 运输计划(二分+树上差分)的更多相关文章

  1. 洛谷 P2680 运输计划-二分+树上差分(边权覆盖)

    P2680 运输计划 题目背景 公元 20442044 年,人类进入了宇宙纪元. 题目描述 公元20442044 年,人类进入了宇宙纪元. L 国有 nn 个星球,还有 n-1n−1 条双向航道,每条 ...

  2. Luogu P2680 运输计划(二分+树上差分)

    P2680 运输计划 题意 题目背景 公元\(2044\)年,人类进入了宇宙纪元. 题目描述 公元\(2044\)年,人类进入了宇宙纪元. \(L\)国有\(n\)个星球,还有\(n-1\)条双向航道 ...

  3. P2680 运输计划 二分+树上差分

    又咕咕了几天\(QwQ\) 思路:二分+树上差分 提交:\(\geq5\)次 错因:\(lca\)写错+卡了很久常数(哪位大佬帮我康康,有更好的写法请指出\(QwQ\)) 题解: 我们先将原问题转化为 ...

  4. 洛谷P2680 运输计划(树上差分+二分)

    传送门 考虑树上乱搞 首先这是满足二分性质的,如果在某个时间可以完成工作那么比他更长的时间肯定也能完成工作 然后考虑二分,设当前答案为$mid$,如果有一条链的长度大于$mid$,那么这条链上必须得删 ...

  5. BZOJ 4326 NOIP2015 运输计划 (二分+树上差分)

    4326: NOIP2015 运输计划 Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1930  Solved: 1231[Submit][Statu ...

  6. NOIP2015Day2T3运输计划(二分+树上差分)

    做了这么多NOIPTG的题,这是唯一 一道一眼秒的T3(有时候T2还不会做QAQ)... 题目大意就不说了QWQ 思路大概是:啊最大值最小化,来个二分.检验mid的话,显然就是用最长路径减去所有边权& ...

  7. 【Luogu】P2680运输计划(树上差分+二分)

    题目链接 总体思路……怎么说呢……是个暴力吧…… 首先用倍增预处理出每条路径的长度. 然后按长度把路径排序. 然后二分答案.对于当前答案mid检验,怎么检验呢? 首先差分把所有长度比mid大的链上除了 ...

  8. [luogu]P2680 运输计划[二分答案][树上差分]

    [luogu]P2680 [NOIP2015]运输计划 题目背景 公元 2044 年,人类进入了宇宙纪元. 题目描述 L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之间,这 n ...

  9. P2680 运输计划[二分+LCA+树上差分]

    题目描述 公元20442044 年,人类进入了宇宙纪元. L 国有 nn 个星球,还有 n-1n−1 条双向航道,每条航道建立在两个星球之间,这 n-1n−1 条航道连通了 LL 国的所有星球. 小 ...

  10. luogu P2680 运输计划 (二分答案+树上差分)

    题目背景 公元 20442044 年,人类进入了宇宙纪元. 题目描述 公元20442044 年,人类进入了宇宙纪元. L 国有 nn 个星球,还有 n-1n−1 条双向航道,每条航道建立在两个星球之间 ...

随机推荐

  1. python类的反射

    反射 通过字符串映射或者修改程序运行时的状态.属性.方法, 有一下4个方法 小例子--根据用户输入调用方法: class Dog(object): def __init__(self,name): s ...

  2. Servlet是线程安全的吗?

    Servlet不是线程安全的. 要解释为什么Servlet为什么不是线程安全的,需要了解Servlet容器(即Tomcat)使如何响应HTTP请求的. 当Tomcat接收到Client的HTTP请求时 ...

  3. shell脚本学习(1)列出一组IP内所有活动主机

    1.用ping命令编写脚本来查询一组IP地址同时检查他们是否处于活动状态 [zhi@centos7 ~]$ vi CheckHostStatus.sh [zhi@centos7 ~]$ ll 总用量 ...

  4. 概念:RPG游戏中两个兵种互相攻击的逻辑

    直接上题目: 解析题目: 根据题目的解析,进行代码的实现: 输出结果: 心得: (1) 当我们面对‘公式结果不是我们想要的’时,应该在脑海里将一个完整的攻击流程进行想象,就会对流程有个更清晰的思路 ( ...

  5. ring0 SSDTHook 实现x64/x86

    #include "HookSSDT.h" #include <ntimage.h> #define SEC_IMAGE 0x001000000 ULONG32 __N ...

  6. ARM实验3 ——串口实验

    uart串口实验 实验内容: 编写UART模块程序,通过串口将信息打印到终端. 实验目的: 熟悉开发环境的使用. 掌握exynos4412处理器的UART功能. 实验平台: FS4412开发板,ecl ...

  7. 利用批处理结合Msbuild实现快速编译

    我们经常在用vs2005做项目的时候会把一个项目分成几个模块(不管是对于功能上,还是系统构架上面),为的是以后部署,还有修改维护时候的方便.这样就会带来一个问题,随着模块的增加(这里所说得每个模块就是 ...

  8. ZOJ - 2112 Dynamic Rankings(BIT套主席树)

    纠结了好久的一道题,以前是用线段树套平衡树二分做的,感觉时间复杂度和分块差不多了... 终于用BIT套函数式线段树了过了,120ms就是快,此题主要是卡内存. 假设离散后有ns个不同的值,递归层数是l ...

  9. hdu-3449 Consumer---有依赖性质的背包

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3449 题目大意: fj打算去买一些东西,在那之前,他需要一些盒子去装他打算要买的不同的物品.每一个盒 ...

  10. 前端高质量知识(一)-JS内存空间详细图解

    变量对象与堆内存   var a = 20;   var b = 'abc';   var c = true;   var d = { m: 20 } 因为JavaScript具有自动垃圾回收机制,所 ...