97: cf 983E 倍增+树套树
$des$
一棵 $n$ 个点的树,树上有 $m$ 条双向的公交线路,每条公交线路都在
两个节点之间沿最短路径往返。
$q$ 次询问从一个点要到达另一个点,在只坐公交的情况下,至少需
要坐几辆公交车;或者判断无法只坐公交到达。
$n,m,q <= 2 \times 10^5$
$sol$
对于每个点,先预处理出从这个点坐一次公交车能最远到达哪个
祖先。对于一条公交线路 (u,v),将 lca 的信息挂在 u,v 上,dfs 一遍向上
更新信息即可。
通过倍增算出从某个点坐 $2^k$ 次最远能到达哪个祖先。这样对于一
条路径 (u,v),我们就能快速算出 u,v 走到 lca 分别至少需要多少辆车,
假设答案分别为 $a,b$,那么询问的答案要么是 $a + b$,要么是 $a + b - 1$
。判断的方法也很简单,先算出 $u$ 向上坐 $a - 1$ 次,$v$ 向上坐 $b - 1$
次,最远能到达哪个节点。若存在一条线路经过这两个节点,则答案
为 $a + b - 1$。也就是判断是否有一条线路的两端分别在这两个节点的
子树中。
这可以变成了一个二维数点问题,时间复杂度 $O((n + m + q)logn)$,不会
$O(nlog^2n)$ 卡过
问题转化为树上存在 $n$ 对点对,每对点对带有不同的权值 a,每次询问点对
$(x, y)$ 二者的子树内是否存在相同的点的权值
可以用线段树套set维护
$code$
#include <bits/stdc++.h> using namespace std; #define gc getchar()
inline int read() {
int x = ; char c = gc;
while(c < '' || c > '') c = gc;
while(c >= '' && c <= '') x = x * + c - '', c = gc;
return x;
} #define E exit(0)
#define Rep(i, a, b) for(int i = a; i <= b; i ++) const int N = 2e5 + ; int fa[N], size[N], topp[N], son[N], deep[N] = {( << )}, lst[N], rst[N], sptime;
int f[N][];
int upst[N]; vector <int> G[N]; int n; void Dfs_1(int u, int f_, int dep) {
fa[u] = f_, deep[u] = dep, size[u] = ;
int S = G[u].size();
Rep(i, , S - ) {
int v = G[u][i];
if(v == f_) continue;
Dfs_1(v, u, dep + );
if(size[v] > size[son[u]]) son[u] = v;
}
} void Dfs_2(int u, int tp) {
topp[u] = tp; lst[u] = ++ sptime;
if(!son[u]) {
rst[u] = sptime; return ;
}
Dfs_2(son[u], tp);
int S = G[u].size();
Rep(i, , S - ) {
int v = G[u][i];
if(v != fa[u] && v != son[u]) Dfs_2(v, v);
}
rst[u] = sptime;
} inline int Lca(int x, int y) {
int tpx = topp[x], tpy = topp[y];
while(tpx != tpy) {
if(deep[tpx] < deep[tpy]) swap(x, y), swap(tpx, tpy);
x = fa[tpx], tpx = topp[x];
}
if(x == y) return x;
if(deep[x] < deep[y]) swap(x, y);
return y;
} void Dfs_3(int u) {
int S = G[u].size();
Rep(i, , S - ) {
int v = G[u][i];
if(v == fa[u]) continue;
Dfs_3(v);
if(deep[upst[v]] < deep[upst[u]]) upst[u] = upst[v];
}
} set <int> Tree[N << ]; #define lson jd << 1
#define rson jd << 1 | 1 void Poi_G(int l, int r, int jd, int x, int num) {
if(Tree[jd].count(num) == ) Tree[jd].insert(num);
if(l == r) return ;
int mid = (l + r) >> ;
if(x <= mid) Poi_G(l, mid, lson, x, num);
else Poi_G(mid + , r, rson, x, num);
} bool flag; set <int> :: iterator wl, wr; void Sec_A(int l, int r, int jd, int x, int y, int fl, int fr) {
if(flag || Tree[jd].size() == ) return ;
if(x <= l && r <= y) {
if(Tree[jd].count(fl) || Tree[jd].count(fr)) {
flag = ; return ;
}
Tree[jd].insert(fr);
wl = Tree[jd].lower_bound(fl);
wr = Tree[jd].lower_bound(fr);
Tree[jd].erase(fr);
if(wr == wl) return ;
if(wr != wl) flag = ;
return ;
}
int mid = (l + r) >> ;
if(x <= mid && flag == ) Sec_A(l, mid, lson, x, y, fl, fr);
if(y > mid && flag == ) Sec_A(mid + , r, rson, x, y, fl, fr);
} inline void Calc(int x, int y) {
int ret;
int lca = Lca(x, y);
int a = , b = , lstx = x, lsty = y, tmpx = x, tmpy = y;
for(int i = ; i >= ; i --) {
if(deep[f[x][i]] > deep[lca]) a += ( << i), x = f[x][i];
}
for(int i = ; i >= ; i --) {
if(deep[f[y][i]] > deep[lca]) b += ( << i), y = f[y][i];
}
if(deep[f[x][]] > deep[lca] || deep[f[y][]] > deep[lca]) {
puts("-1"); return ;
}
flag = ;
int l1 = lst[x], r1 = rst[x], l2 = lst[y], r2 = rst[y];
Sec_A(, n, , l1, r1, l2, r2); Sec_A(, n, , l2, r2, l1, r1);
if(lca == x || lca == y) ret = a + b + ;
else if(flag == ) ret = a + b + ;
else ret = a + b + ;
printf("%d\n", ret);
} int main() {
n = read(); Rep(i, , n) {
int u = read();
G[u].push_back(i), G[i].push_back(u);
} Dfs_1(, , );
Dfs_2(, ); int m = read();
Rep(i, , m) {
int u = read(), v = read();
int lca = Lca(u, v);
if(deep[lca] < deep[upst[u]] || upst[u] == ) upst[u] = lca;
if(deep[lca] < deep[upst[v]] || upst[v] == ) upst[v] = lca;
Poi_G(, n, , lst[u], lst[v]);
} Dfs_3();
Rep(i, , n) f[i][] = upst[i];
Rep(i, , ) {
Rep(j, , n) f[j][i] = f[f[j][i - ]][i - ];
} m = read();
Rep(i, , m) {
int a = read(), b = read();
Calc(a, b);
}
return ;
}
97: cf 983E 倍增+树套树的更多相关文章
- BZOJ 3110: [Zjoi2013]K大数查询 [树套树]
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6050 Solved: 2007[Submit][Sta ...
- BZOJ4170 极光(CDQ分治 或 树套树)
传送门 BZOJ上的题目没有题面-- [样例输入] 3 5 2 4 3 Query 2 2 Modify 1 3 Query 2 2 Modify 1 2 Query 1 1 [样例输出] 2 3 3 ...
- bzoj3262: 陌上花开(树套树)
#include <iostream> #include <cstdio> #include <cstring> #include <cmath> #i ...
- bzoj3295: [Cqoi2011]动态逆序对(树套树)
#include <iostream> #include <cstdio> #include <cstring> #include <cmath> #i ...
- BZOJ 3110 k大数查询 & 树套树
题意: 有n个位置,每个位置可以看做一个集合,现在要求你实现一个数据结构支持以下功能: 1:在a-b的集合中插入一个数 2:询问a-b集合中所有元素的第k大. SOL: 调得火大! 李建说数据结构题能 ...
- BZOJ 3110 树套树 && 永久化标记
感觉树套树是个非常高深的数据结构.从来没写过 #include <iostream> #include <cstdio> #include <algorithm> ...
- 【BZOJ】1901: Zju2112 Dynamic Rankings(区间第k小+树套树)
http://www.lydsy.com/JudgeOnline/problem.php?id=1901 这题调了我相当长的时间,1wa1a,我是第一次写树套树,这个是树状数组套splay,在每个区间 ...
- hdu 4417 Super Mario/树套树
原题链接:http://acm.hdu.edu.cn/showproblem.php?pid=4417 题意很简单,给定一个序列求一个区间 [L, R,]中小于等于H的元素的个数. 好像函数式线段树可 ...
- Uva 3767 Dynamic len(set(a[L:R])) 树套树
Dynamic len(set(a[L:R])) Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 https://uva.onlinejudge.org/in ...
随机推荐
- Spring Cloud--Hystrix服务熔断(线程隔离/服务降级)代码实现
一旦服务阻塞就进行服务降级或线程隔离.要不然就会导致大面积服务的瘫痪,Hystrix就是干这个的,一出现不健康的服务就进行熔断,不阻塞后面线程的执行. 引入依赖: 加注解: 这三个注解可以用一个注解搞 ...
- SpringBoot与整合其他技术
SpringBoot与整合其他技术 5.1 SpringBoot整合Mybatis 5.1.1 添加Mybatis的起步依赖 <!--mybatis起步依赖--> <dependen ...
- 使用CertUtil.exe下载远程文件
使用CertUtil.exe下载远程文件 证书 https://www.cnblogs.com/17bdw/p/8728656.html 1.前言 经过国外文章信息,CertUtil.exe下载恶意软 ...
- OpenWrt 中查看 Flash RAM CPU 信息
OpenWrt 中查看 Flash RAM CPU 信息 来源 https://blog.csdn.net/mcusun2000/article/details/51130434 硬件: QCA95 ...
- 深入理解JVM(一) -- 自动内存管理机制
Java运行时数据区域分为:程序计数器,虚拟机栈,本地方法栈,Java堆,方法区,运行时常量池,直接内存,结构如下: 1.程序计数器: 是一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示 ...
- SpringMVC、SpringFox和Swagger整合项目实例
目标 在做项目的时候,有时候需要提供其它平台(如业务平台)相关的HTTP接口,业务平台则通过开放的HTTP接口获取相关的内容,并完成自身业务~ 提供对外开放HTTP API接口,比较常用的是采用Spr ...
- ECSHOP(3.0.0升级3.6.0)帮助教程
说明: 本文档只针对于未做过二开的ECSHOP3.0 用户 1.准备材料 先确保正在使用的ECShop系统版本为ecshop3.0.0并且代码没有经过二次开发,然后下载最新的ECShop3.6.0安装 ...
- redis 异常 MISCONF Redis is configured to save RDB snapshots, but is currently not able to persist on disk
MISCONF Redis is configured to save RDB snapshots, but is currently not able to persist on disk. 解决方 ...
- Linux学习django-CentOS部署自己本地的django项目
前言 自己本地写好的django项目,如何部署到linux服务器上,让其他的小伙伴也能访问呢?本篇以centos系统为例,把本地写好的django项目部署到linux服务器上环境准备: 环境准备:1. ...
- Altium designer 如何将2D PCB转换成3D
点击键盘数字键的3,即可,2键可以切换回2D效果,但是如果要看元器件的3D效果,那么元器件封装必须带有3D模型才行! 先按3切换到三维界面,然后按住shift不放,按鼠标右键调整视图角度.