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 ...
随机推荐
- 物流管理系统(SSM+vue+shiro)【前后台】
一.简单介绍项目 该项目是属于毕业设计项目之一,有前台的用户下单.有司机进行接单.有管理员进行操作后台,直接进入主题 毕设.定制开发 联系QQ:761273133 登录主页: 手机号码+验证码登录 或 ...
- mysql学习之基础篇08 UTF8编码
这次我们来说一下在Mysql中的编码问题: 我们知道应用于计算机的最早的字符集是ASCII,它所组成的编码是ASCII编码:由于对于其他国家来说它所容纳的字符个数比较少,后来就出现了ANSI字符集,它 ...
- Nginx配置文件示例
Nginx的配置文件示例:(仅供参考) 强烈建议先将默认的配置文件备份再进行操作! 请根据自己项目的实际路径来配置相关路径! uwsgi配置文件请参考:uwsgi配置文件示例 # For more i ...
- [nginx] nginx源码分析--proxy模式下nginx的自动重定向auto_redirect
描述 我们配置了一个proxy模式下的nginx, upstream backend-test { server ; } server { listen ; location = /nginx/hww ...
- [ipsec][strongswan] strongswan源码分析-- (三) xfrm与strongswan内核接口分析
目录 strongwan sa分析(三) xfrm与strongswan内核接口分析 1. strongswan的实现 2. 交互机制 4. xfrm的消息通信的实现 strongwan sa分析(三 ...
- 定时任务at
1. at一次性执行定时任务 1.1 at服务管理与访问控制 at命令要想正确执行,需要atd服务的支持.atd服务是独立的服务 [root@centos2 ~]# systemctl status ...
- linux下补丁制作和使用方法
两个文件的情况: 制作补丁: $ diff test1.c test2.c > test.patch 给test1.c打补丁: $ patch test1.c < test.patch 还 ...
- ISCC之web1
由题意知,爆破可行. 简单说一下抓包过程,第一个抓到的POST包直接放掉,右键扫描站点,扫出来第二个POST包,第二个POST包cookie,验证码,密码均为空,于是我开始尝试去利用, 经过几次rep ...
- Windows性能计数器监控实践
Windows性能计数器(Performance Counter)是Windows提供的一种系统功能,它能实时采集.分析系统内的应用程序.服务.驱动程序等的性能数据,以此来分析系统的瓶颈.监控组件的表 ...
- 小顶堆第二弹-----堆降序排序(C语言非递归)
现在po一下C语言版本的,留作以后接口使用. 1 #include <stdio.h> #include <stdlib.h> #define HEAP_SIZE 100 #d ...