题目链接  BZOJ4326

这个程序在洛谷上TLE了……惨遭卡常

在NOIP赛场上估计只能拿到95分吧= =

把边权转化成点权

首先求出每一条路径的长度

考虑二分答案,$check(now)$

对于当前那些长度大于$now$的路径,用差分求出这些路径经过的点的次数

设这些路径条数为l, 长度最大的路径减去$now$的值为$mx$

(点$x1$经过$y1$次,点$x2$经过$y2$次,..., 点$xm$经过$ym$次)

如果我们能找到一条边(一个点),满足$xk = l$ 且 $yk >= mx$ 则$check$成功,否则失败。

#include <bits/stdc++.h>

using namespace std;

#define rep(i, a, b)    for (int i(a); i <= (b); ++i)
#define dec(i, a, b) for (int i(a); i >= (b); --i)
#define MP make_pair
#define fi first
#define se second typedef long long LL;
typedef pair <int, int> PII; const int N = 3e5 + 10; struct node{
int x, y, lca, w;
void scan(){ scanf("%d%d", &x, &y);}
void print(){ printf("%d %d %d %d\n", x, y, lca, w);}
} path[N]; int father[N], deep[N], sz[N], son[N], top[N];
int a[N], b[N], f[N], g[N];
int n, m, x, y, z, l, r, ans;
vector <PII> v[N]; void dfs(int x, int fa, int dep, int now){
sz[x] = 1;
deep[x] = dep;
father[x] = fa;
a[x] = now;
b[x] = b[fa] + a[x];
int ct = (int)v[x].size();
rep(i, 0, ct - 1){
int u = v[x][i].fi;
if (u == fa) continue;
dfs(u, x, dep + 1, v[x][i].se);
sz[x] += sz[u];
if (sz[son[x]] < sz[u]) son[x] = u;
}
} void dfs2(int x, int fa, int tp){
top[x] = tp;
if (son[x]) dfs2(son[x], x, tp);
int ct = (int)v[x].size();
rep(i, 0, ct - 1){
int u = v[x][i].fi;
if (u == son[x] || u == fa) continue;
dfs2(u, x, u);
}
} void calc(int x, int fa){
int ct = (int)v[x].size();
rep(i, 0, ct - 1){
int u = v[x][i].fi;
if (u == fa) continue;
calc(u, x);
f[x] += f[u];
}
} int LCA(int x, int y){
for (; top[x] ^ top[y]; ){
if (deep[top[x]] < deep[top[y]]) swap(x, y);
x = father[top[x]];
} return deep[x] > deep[y] ? y : x;
} bool check(int now){
int cnt = 0;
int mx = 0;
memset(f, 0, sizeof f);
rep(i, 1, m) if (path[i].w > now){
int x = path[i].x, y = path[i].y, w = path[i].w, lca = path[i].lca;
++cnt;
if (lca == y) ++f[x], --f[y];
else ++f[x], ++f[y], f[lca] -= 2;
mx = max(mx, w - now);
} calc(1, 0);
rep(i, 1, n) if (a[i] >= mx && f[i] == cnt) return true;
return false;
} int main(){ scanf("%d%d", &n, &m);
rep(i, 2, n){
scanf("%d%d%d", &x, &y, &z);
v[x].push_back(MP(y, z));
v[y].push_back(MP(x, z));
} dfs(1, 0, 0, 0);
dfs2(1, 0, 1); rep(i, 1, m){
path[i].scan();
if (deep[path[i].x] < deep[path[i].y]) swap(path[i].x, path[i].y);
path[i].lca = LCA(path[i].x, path[i].y);
path[i].w = b[path[i].x] + b[path[i].y] - 2 * b[path[i].lca];
} l = 0, r = 3e8;
while (l + 1 < r){
int mid = l + r >> 1;
if (check(mid)) r = mid; else l = mid + 1;
} if (check(l)) ans = l; else ans = r;
printf("%d\n", ans);
return 0;
}

BZOJ 4326 NOIP2015 运输计划(二分答案 + 树上差分思想)的更多相关文章

  1. BZOJ 4326: NOIP2015 运输计划(二分,树上差分)

    Time Limit: 30 Sec  Memory Limit: 128 MBSubmit: 1945  Solved: 1243[Submit][Status][Discuss] Descript ...

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

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

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

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

  4. loj2425 「NOIP2015」运输计划[二分答案+树上差分]

    看到题意最小化最长路径,显然二分答案,枚举链长度不超过$\text{mid}$,然后尝试检验.````` 检验是否存在这样一个边置为0后,全部链长$\le\text{mid}$,其最终目标就是.要让所 ...

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

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

  6. BZOJ 4326 NOIP2015 运输计划(树上差分+LCA+二分答案)

    4326: NOIP2015 运输计划 Time Limit: 30 Sec  Memory Limit: 128 MB Submit: 1388  Solved: 860 [Submit][Stat ...

  7. bzoj 4326: NOIP2015 运输计划【树链剖分+二分+树上差分】

    常数巨大,lg上开o2才能A 首先预处理出运输计划的长度len和lca,然后二分一个长度w,对于长度大于w的运输计划,在树上差分(d[u]+1,d[v]+1,d[lca]-2),然后dfs,找出所有覆 ...

  8. bzoj 4326: NOIP2015 运输计划

    4326: NOIP2015 运输计划 Time Limit: 30 Sec Memory Limit: 128 MB Description 公元 2044 年,人类进入了宇宙纪元.L 国有 n 个 ...

  9. bzoj 4326: NOIP2015 运输计划(二分+树链剖分)

    传送门 题解: 树链剖分快速求解任意两点间的路径的权值和: 然后,二分答案: 此题的难点是如何快速求解重合路径? 差分数组可以否??? 在此之前先介绍一下相关变量: int fa[maxn]; int ...

随机推荐

  1. 私有DockerHub搭建

    docker简介 一个开源的应用容器引擎,可以用来打包程序,可以包入依赖环境,这样只需要提供docker image即可,类似于虚拟机,但是更轻量级. 几个概念: Paas,platform as a ...

  2. Java代码中的(解压7z加密版)

    maven:需要加上这个下载这两个包 <dependency> <groupId>net.sf.sevenzipjbinding</groupId> <art ...

  3. UVALive - 8273 Assigning Frequencies (搜索 )

    n个点的一张图,问能否给每个点染上三种颜色中的一种,使得没有相邻的点颜色相同? n <= 35. Sample Input 4 6 6 0 3 1 5 3 2 2 5 0 4 1 0 7 12 ...

  4. jmeter throughput controller

    工作方式:可以按规定次数执行,也可以选择按百分比执行,其中的百分比必须是10,20,30类似的整数. 使用场景:可以随机的去按百分比浏览网址. 以下是具体脚本:

  5. visual studio 2019安装秘钥

    美国时间4.2微软发布了最新版本的visual studio 2019 现在贴出visual studio2019的秘钥,有需要的请自取: Visual Studio 2019 Enterprise( ...

  6. 算法学习记录-查找——二叉排序树(Binary Sort Tree)

    二叉排序树 也称为 二叉查找数. 它具有以下性质: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值. 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值. 它的左.右子树也分别 ...

  7. PHP中文网 学习阶段规划

    1.第一阶段: 前端基础 前端基础课程大纲 教学内容 教学重点 1.HTML5 HTML简介.HTML标签详解.字符编码的奥秘.Html5新特性与常用标签 2.CSS3 CSS简介.CSS的引入方式. ...

  8. 求1+2+...+n 【微软面试100题 第十二题】

    题目要求: 要求不能使用乘除法,for/while/if/else/switch/case等关键字以及条件判断语句(A?B:C). 参考资料:剑指offer第46题 题目分析: 方法1:利用类的静态成 ...

  9. TOJ1550: Fiber Communications

    1550: Fiber Communications  Time Limit(Common/Java):1000MS/10000MS     Memory Limit:65536KByteTotal ...

  10. Farey sequences

    n阶的法里数列是0和1之间最简分数的数列,由小至大排列,每个分数的分母不大于n. Stern-Brocot树(SB Tree)可以生成这个序列 {0/1,1/1} {0/1,1/2,1/1} {0/1 ...