一棵带边权的树,多次询问 $x$ 到编号为 $[l,r]$ 的点最短距离是多少

$n \leq 100000$

sol:

动态点分治,每层重心维护到所有点的距离

查询的时候在管辖这个点的 log 层线段树里查就可以了

因为这样每一层的答案只会漏而不会错,所以正确性有保障

不会写点分治了...orz

#include <bits/stdc++.h>
#define LL long long
#define rep(i, s, t) for (register int i = (s), i##end = (t); i <= i##end; ++i)
#define dwn(i, s, t) for (register int i = (s), i##end = (t); i >= i##end; --i)
using namespace std;
inline int read() {
int x = , f = ;
char ch;
for (ch = getchar(); !isdigit(ch); ch = getchar())
if (ch == '-')
f = -f;
for (; isdigit(ch); ch = getchar()) x = * x + ch - '';
return x * f;
}
const int maxn = , inf = 1e9;
int n, m;
int first[maxn], to[maxn << ], nx[maxn << ], val[maxn], cnt;
inline void add(int u, int v, int w) {
to[++cnt] = v;
val[cnt] = w;
nx[cnt] = first[u];
first[u] = cnt;
}
int sig, root, size[maxn], pa[maxn], f[maxn], vis[maxn];
inline void findroot(int x, int fa) {
size[x] = ;
f[x] = ;
for (int i = first[x]; i; i = nx[i]) {
if ((to[i] == fa) || (vis[to[i]]))
continue;
findroot(to[i], x);
size[x] += size[to[i]];
f[x] = max(f[x], size[to[i]]);
}
f[x] = max(f[x], sig - size[x]);
if (f[x] < f[root])
root = x;
}
inline void build(int x) {
vis[x] = ;
for (int i = first[x]; i; i = nx[i]) {
if (vis[to[i]])
continue;
root = ;
sig = size[to[i]];
findroot(to[i], x);
pa[root] = x;
build(root);
}
}
namespace sp_lca {
int size[maxn], dep[maxn], fa[maxn], bl[maxn], dp[maxn];
inline void dfs1(int x) {
size[x] = ;
for (int i = first[x]; i; i = nx[i]) {
if (to[i] == fa[x])
continue;
fa[to[i]] = x;
dep[to[i]] = dep[x] + ;
dp[to[i]] = dp[x] + val[i];
dfs1(to[i]);
size[x] += size[to[i]];
}
}
inline void dfs2(int x, int col) {
int k = ;
bl[x] = col;
for (int i = first[x]; i; i = nx[i])
if (to[i] != fa[x] && size[to[i]] > size[k])
k = to[i];
if (!k)
return;
dfs2(k, col);
for (int i = first[x]; i; i = nx[i])
if (to[i] != fa[x] && to[i] != k)
dfs2(to[i], to[i]);
}
inline void init() {
dfs1();
dfs2(, );
}
inline int lca(int x, int y) {
while (bl[x] != bl[y]) {
if (dep[bl[x]] < dep[bl[y]])
swap(x, y);
x = fa[bl[x]];
}
return dep[x] < dep[y] ? x : y;
}
} // namespace sp_lca
inline int dis(int x, int y) { return sp_lca::dp[x] + sp_lca::dp[y] - (sp_lca::dp[sp_lca::lca(x, y)] << ); }
int rt[maxn], ls[maxn << ], rs[maxn << ], va[maxn << ], ToT;
inline void Insert(int &x, int l, int r, int pos, int v) {
if (!x)
x = ++ToT, va[x] = inf;
va[x] = min(va[x], v);
if (l == r)
return;
int mid = (l + r) >> ;
if (pos <= mid)
Insert(ls[x], l, mid, pos, v);
else
Insert(rs[x], mid + , r, pos, v);
}
inline int query(int x, int l, int r, int L, int R) {
if (!x)
return inf;
if (L <= l && r <= R)
return va[x];
int mid = (l + r) >> , res = inf;
if (L <= mid)
res = min(res, query(ls[x], l, mid, L, R));
if (R > mid)
res = min(res, query(rs[x], mid + , r, L, R));
return res;
}
int main() {
// freopen("a.in","r",stdin);
// freopen("a.out","w",stdout);
n = read();
rep(i, , n) {
int u = read(), v = read(), w = read();
add(u, v, w);
add(v, u, w);
}
f[] = inf;
root = ;
sig = n;
findroot(, );
build(root);
sp_lca ::init();
rep(i, , n) for (int x = i; x; x = pa[x]) Insert(rt[x], , n, i, dis(x, i));
// rep(i, 1, n) cout << pa[i] << endl;
int q = read();
while (q--) {
int u = read(), v = read(), x = read();
int cur = x, ans = inf;
while (cur) {
ans = min(ans, query(rt[cur], , n, u, v) + dis(cur, x));
cur = pa[cur];
}
cout << ans << endl;
}
}

「2017 山东三轮集训 Day7」Easy的更多相关文章

  1. 【loj6145】「2017 山东三轮集训 Day7」Easy 动态点分治+线段树

    题目描述 给你一棵 $n$ 个点的树,边有边权.$m$ 次询问,每次给出 $l$ .$r$ .$x$ ,求 $\text{Min}_{i=l}^r\text{dis}(i,x)$ . $n,m\le ...

  2. #6145. 「2017 山东三轮集训 Day7」Easy 动态点分治

    \(\color{#0066ff}{题目描述}\) JOHNKRAM 最近在参加 C_SUNSHINE 举办的聚会. C 国一共有 n 座城市,这些城市由 n−1 条无向道路连接.任意两座城市之间有且 ...

  3. LOJ #6145. 「2017 山东三轮集训 Day7」Easy 点分树+线段树

    这个就比较简单了~ Code: #include <cstdio> #include <algorithm> #define N 100004 #define inf 1000 ...

  4. 「2017 山东三轮集训 Day7 解题报告

    「2017 山东三轮集训 Day7」Easy 练习一下动态点分 每个点开一个线段树维护子树到它的距离 然后随便查询一下就可以了 注意线段树开大点... Code: #include <cstdi ...

  5. 「2017 山东三轮集训 Day1」Flair

    模拟赛的题 好神仙啊 题面在这里 之前的Solution很蠢 现在已经update.... 题意 有$ n$个商品价格均为$ 1$,您有$ m$种面值的货币,面值为$ C_1..C_m$ 每种物品你有 ...

  6. 【loj6142】「2017 山东三轮集训 Day6」A 结论题+Lucas定理

    题解: 当奇数 发现答案就是C(n,1)^2+C(n,3)^2+...C(n,n)^2 倒序相加,发现就是C(2n,n) 所以答案就是C(2n,n)/2 当偶数 好像并不会证 打表出来可以得到 2.当 ...

  7. loj #6138. 「2017 山东三轮集训 Day4」Right

    题目: 题解: 暴力一波 \(SG\) 函数可以发现这么一个规律: \(p\) 为奇数的时候 : \(SG(n) = n \% 2\) \(p\) 为偶数的时候 : \(SG(n) = n \% (p ...

  8. loj #6136. 「2017 山东三轮集训 Day4」Left

    题目: 题解: 我们可以发现所有的交换器都是一个位置连接着下一层左侧的排序网络,另一个位置连着另一侧的排序网络. 而下一层是由两个更低阶的排序网络构成的. 两个网络互不干扰.所以我们可以通过第一行和最 ...

  9. Loj #6142. 「2017 山东三轮集训 Day6」A

    link: https://loj.ac/problem/6142 推完一波式子之后发现求的是:ΣC(N,i)^2, 其中i是偶数. 然后就可以卢卡斯乱搞了,分奇偶和之前的答案合并就好了233. #i ...

随机推荐

  1. yii2-lock-form 也许这就是你想要的,阻止表单多次提交

    是不是被用户的行为所困扰? 一个表单用户点击提交按钮了N次,这也导致了数据提交了N次. 为了此受到了测试的欺辱,受到了老板的批评? 不用怕,它就是来拯救你的. 第一步:打开命令行,敲入 compose ...

  2. PAT 天梯赛 L1-043. 阅览室 【STL】

    题目链接 https://www.patest.cn/contests/gplt/L1-043 思路 将每一次 借出和归还 都用 MAP 标记 如果归还的时候 已经被标记过了 那么 ANS ++ 并且 ...

  3. $2015 武汉森果公司web后端开发实习日记----书写是为了更好的思考

    找暑期实习,3月份分别投了百度和腾讯的实习简历,都止步于笔试,总结的主要原因有两点:基础知识不扎实,缺乏项目经验.后来到拉勾网等网站上寻找实习,看了很多家,都还是处于观望状态.后来参加了武汉实习吧在大 ...

  4. I.MX6Q(TQIMX6Q/TQE9)学习笔记——U-Boot移植

    其实Freescale的BSP移植文档已经将u-boot的移植步骤讲述的非常详细了,但为了以后方便查阅,还是按照自己的理解记录在这里. 获取源码 根据前一篇文章搭建好LTIB环境后就可以非常方便的导出 ...

  5. C语言:内存字节对齐详解

    转:http://blog.csdn.net/arethe/article/details/2548867 一.什么是对齐,以及为什么要对齐: 1. 现代计算机中内存空间都是按照byte划分的,从理论 ...

  6. python的计算保留小数

    1.要使得算术运算的结果有小数,则运算的对象至少有一个是float型的. 2.控制小数的位数:字符串格式化 格式:需要进行格式化的字符串%插入对象 需要进行格式化的字符串中带有一个或多个嵌入的转换目标 ...

  7. 转载:ensemble计划和数据库

    原文来源:x2yline在生信进化树上的评论,http://www.biotrainee.com/thread-626-1-1.html Ensemble( ensembl.org网站是常用真核生物参 ...

  8. INSPIRED启示录 读书笔记 - 第24章 平滑部署

    避免更新产品导致用户反感 毫无征兆地更新不必要的版本会令用户产生反感.不是所有用户都喜欢新版本的产品.用户产生反感主要有几个原因 1.事前没有收到更新通知,用户觉得措手不及 2.用户没时间学习.适应新 ...

  9. 如何判断Linux服务器是否被入侵?

    被入侵服务器的症状 当服务器被没有经验攻击者或者自动攻击程序入侵了的话,他们往往会消耗 100% 的资源.他们可能消耗 CPU 资源来进行数字货币的采矿或者发送垃圾邮件,也可能消耗带宽来发动 DoS ...

  10. 【codevs2333】&【BZOJ2002】弹飞绵羊[HNOI2010](分块)

    我其实是在codevs上看到它的题号后才去做这道题的...2333... 题目传送门:codevs:http://codevs.cn/problem/2333/ bzoj:http://www.lyd ...