【洛谷 P4211】[LNOI2014]LCA(树链剖分,差分)
看到题目肯定首先想到要求LCA(其实是我菜),可乍一看,n与q的规模为5W,
求LCA的复杂度为\(O(logN)\),那么总时间复杂度为\(O(nq\ log\ n)\)。
怎么搞呢?
会树上差分的都知道,要对一条链进行操作,比如说链上的节点权值\(+p\),就要对两个端点分别\(+p\),然后对\(LCA\)及其父亲分别\(-p\)。
和这个思想差不多,设两个点为\(u, v\),那么求\(dep(LCA(u,v))\)只需要把\((root,u)\)之前的路径所有点权值\(+1\),然后统计\((root,v)\)路径上的权值和就行了,具体请自行脑补我觉得很好理解啊,这个过程显然可以用树剖来完成。
然后就是差分,注意不是树上差分,\(\sum_{i=l}^{r}LCA(i,z)=\sum_{i=1}^{r}LCA(i,z)-\sum_{i=1}^{l-1}LCA(i,z)\)
考虑离线,设每个询问为\((l,r,z)\),定义一个结构体,包含\(x\)(端点),\(k\)(-1为左端点,1为右端点),\(z\)(要求LCA的点),\(id\)(是哪个询问),用2个空间分别存储左端点\(l\)和右端点\(r\),然后按端点编号排序。此时存放询问的数组应该是有\(2q\)个的。定义一个指针\(cur\),表示当前已经把前\(cur\)个点到根的路径\(+1\)了。枚举\(i(1~2q)\),如果是左端点,把\(cur\)更新到\(x-1\),(\(while(cur<x-1) Updata(1,++cur);\)),把\(ans[id]\)减去\(z\)到根的权值和,相当于\(-\sum_{i=1}^{l-1}LCA(i,z)\)。如果是右端点,同理,把\(cur\)更新到\(x\),然后\(ans[id]\)加上...这题就愉快地切掉了。时间复杂度\(O(n\ log^2\ n)\)
手写树剖一遍过,好激动
#include <cstdio>
#include <algorithm>
using std::swap;
using std::sort;
#define left (now << 1)
#define right (now << 1 | 1)
const int MAXN = 50010;
const int MOD = 201314;
struct Edge{
int next, to;
}e[MAXN];
int head[MAXN], num, n, q, a, b, c;
int fa[MAXN], maxson[MAXN], size[MAXN], top[MAXN], dfsx[MAXN], ID, pos[MAXN], dep[MAXN], ans[MAXN];
inline void Add(int from, int to){
e[++num].to = to;
e[num].next = head[from];
head[from] = num;
}
int sum[MAXN << 2], lazy[MAXN << 2];
inline void pushdown(int now, int len){
if(lazy[now]){
sum[left] += lazy[now] * (len - (len >> 1));
sum[right] += lazy[now] * (len >> 1);
lazy[left] += lazy[now];
lazy[right] += lazy[now];
lazy[now] = 0;
}
}
inline void pushup(int now){
sum[now] = sum[left] + sum[right];
}
/*void Build(int now, int l, int r){
if(l != r){
int mid = (l + r) >> 1;
Build(left, l, mid);
Build(right, mid + 1, r);
pushup(now);
}
else sum[now] = a[l];
}*/
void Change(int now, int l, int r, int wl, int wr, int p){
if(l > wr || r < wl) return ;
if(l >= wl && r <= wr){
sum[now] += p * (r - l + 1); lazy[now] += p; return ;
}
pushdown(now, r - l + 1);
int mid = (l + r) >> 1;
Change(left, l, mid, wl, wr, p);
Change(right, mid + 1, r, wl, wr, p);
pushup(now);
}
int Query(int now, int l, int r, int wl, int wr){
if(l > wr || r < wl) return 0;
if(l >= wl && r <= wr) return sum[now];
pushdown(now, r - l + 1);
int ans = 0;
int mid = (l + r) >> 1;
ans += Query(left, l, mid, wl, wr);
ans += Query(right, mid + 1, r, wl, wr);
return ans;
}
struct ASK{
int x, z, k, id;
bool operator < (const ASK &A) const{ //按照端点排序
return x == A.x ? k < A.k : x < A.x;
}
}Ans[MAXN << 1];
void dfs1(int u){
dep[u] = dep[fa[u]] + 1;
size[u] = 1;
for(int i = head[u]; i; i = e[i].next){
dfs1(e[i].to);
size[u] += size[e[i].to];
if(size[e[i].to] > size[maxson[u]])
maxson[u] = e[i].to;
}
}
void dfs2(int u, int t){
dfsx[++ID] = u;
top[u] = t;
if(maxson[u]) dfs2(maxson[u], t);
for(int i = head[u]; i; i = e[i].next){
if(e[i].to != maxson[u])
dfs2(e[i].to, e[i].to);
}
}
void Updata(int u, int v, int p){
int x = top[u], y = top[v];
while(x != y){
if(dep[x] > dep[y]) swap(x, y), swap(u, v);
Change(1, 1, n, pos[y], pos[v], p);
u = fa[x]; x = top[u];
v = fa[y]; y = top[v];
}
if(dep[u] > dep[v]) swap(u, v);
Change(1, 1, n, pos[u], pos[v], p);
}
int Solve(int u, int v){
int ans = 0;
int x = top[u], y = top[v];
while(x != y){
if(dep[x] > dep[y]) swap(x, y), swap(u, v);
ans += Query(1, 1, n, pos[y], pos[v]);
u = fa[x]; x = top[u];
v = fa[y]; y = top[v];
}
if(dep[u] > dep[v]) swap(u, v);
ans += Query(1, 1, n, pos[u], pos[v]);
return ans;
}
int cur;
int main(){
scanf("%d%d", &n, &q);
for(int i = 2; i <= n; ++i)
scanf("%d", &fa[i]), Add(fa[i] += 1, i);
for(int i = 1; i <= q; ++i){
scanf("%d%d%d", &a, &b, &c);
Ans[i].x = a + 1;
Ans[i + q].x = b + 1;
Ans[i].z = Ans[i + q].z = c + 1;
Ans[i].k = -1;
Ans[i + q].k = 1;
Ans[i].id = Ans[i + q].id = i;
}
dfs1(1);
dfs2(1, 1);
for(int i = 1; i <= n; ++i)
pos[dfsx[i]] = i;
sort(Ans + 1, Ans + q * 2 + 1);
for(int i = 1; i <= 2 * q; ++i){
if(Ans[i].k < 0) //是左端点
while(cur < Ans[i].x - 1)
Updata(1, ++cur, 1);
else //是右端点
while(cur < Ans[i].x)
Updata(1, ++cur, 1);
ans[Ans[i].id] += Ans[i].k * Solve(1, Ans[i].z);
}
for(int i = 1; i <= q; ++i)
printf("%d\n", (ans[i] + MOD) % MOD);
return 0;
}
【洛谷 P4211】[LNOI2014]LCA(树链剖分,差分)的更多相关文章
- 洛谷 P4211 [LNOI2014]LCA (树链剖分+离线)
题目:https://www.luogu.org/problemnew/solution/P4211 相当难的一道题,其思想难以用言语表达透彻. 对于每个查询,区间[L,R]中的每个点与z的lca肯定 ...
- 洛谷$P4211\ [LNOI2014]\ LCA$ 树链剖分+线段树
正解:树剖+线段树 解题报告: 传送门$QwQ$ 看到$dep[lca]$啥的就想到之前托腮腮$CSP$模拟$D1T3$的那个套路,,, 然后试下这个想法,于是$dep[lca(x,y)]=\sum_ ...
- [BZOJ3626] [LNOI2014]LCA(树链剖分)
[BZOJ3626] [LNOI2014]LCA(树链剖分) 题面 给出一棵N个点的树,要求支持Q次询问,每次询问一个点z与编号为区间[l,r]内的点分别求最近公共祖先得到的最近公共祖先深度和.N, ...
- BZOJ 3626: [LNOI2014]LCA [树链剖分 离线|主席树]
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2050 Solved: 817[Submit][Status ...
- BZOJ 3626: [LNOI2014]LCA( 树链剖分 + 离线 )
说多了都是泪啊...调了这么久.. 离线可以搞 , 树链剖分就OK了... -------------------------------------------------------------- ...
- BZOJ2243 洛谷2486 [SDOI2011]染色 树链剖分
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ2243 题目传送门 - 洛谷2486 题意概括 一棵树,共n个节点. 让你支持以下两种操作,共m次操 ...
- BZOJ3626[LNOI2014]LCA——树链剖分+线段树
题目描述 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1.设dep[i]表示点i的深度,LCA(i,j)表示i与j的最近公共祖先.有q次询问,每次询 ...
- bzoj 3626 : [LNOI2014]LCA (树链剖分+线段树)
Description 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1.设dep[i]表示点i的深度,LCA(i,j)表示i与j的最近公共祖先.有q ...
- 【bzoj3626】[LNOI2014]LCA 树链剖分+线段树
题目描述 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1.设dep[i]表示点i的深度,LCA(i,j)表示i与j的最近公共祖先.有q次询问,每次询 ...
- 洛谷P3313 [SDOI2014]旅行(树链剖分 动态开节点线段树)
题意 题目链接 Sol 树链剖分板子 + 动态开节点线段树板子 #include<bits/stdc++.h> #define Pair pair<int, int> #def ...
随机推荐
- tp5.0 模型查询数据的返回类型,分页
一开始用painate()这个函数的时候,发现有的查询方式不能使用这个函数,由此了解到了模型查询和普通查询返回类型的不同 1.原生查询方法 Db::query("select * from ...
- mingw编译opencv2.4.13问题记录
为了在程序中用regex,升级了我的mingw,结果官网上的GCC版本都到6.3了,之前一直用4.8.换了编译器以后,对opencv2.4.10的引用就出了问题:undefined reference ...
- 语法测试cnblogs使用Markdown
参考自作业部落Cmd Markdown 编辑器 https://www.zybuluo.com 欢迎使用 Cmd Markdown 编辑阅读器 什么是 Markdown Markdown 是一种方便记 ...
- await和async再学习
await太不容易理解了,自己常常迷惑,不知道该怎么用. 文章:探索c#之Async.Await剖析 这篇文章,有一个很清晰的描述: 使用Async标记方法Async1为异步方法,用Await标记Ge ...
- oop &&GP 模板 ---> 特化和偏特化
OOP面向对象编程 GP泛型编程(generic programming) 两者的主要区别就是OOP将数据和对数据的操作放在一起, GP就是将数据和操作独立开来 GP: 数据就是container ...
- 基于SDN的IP RAN网络虚拟化技术
http://www.zte.com.cn/cndata/magazine/zte_technologies/2014/2014_4/magazine/201404/t20140421_422858. ...
- 【工具学习】——Maven的安装与配置
[含义] 什么是构建? 构建,英文build.构建包括编译.运行.生成文档.打包.部署等等工作内容,如果我们每天手工去干这些事情,那会浪费很多的时间.因此,构建管理工具应运而生. maven,作为项目 ...
- Android 实现异步加载图片
麦洛开通博客以来,有一段时间没有更新博文了.主要是麦洛这段时间因项目开发实在太忙了.今天周六还在公司加班,苦逼程序猿都是这样生活的. 今天在做项目的时候,有一个实现异步加载图片的功能,虽然比较简单但还 ...
- 大数据Hadoop-2
大数据Hadoop学习之搭建Hadoop平台(2.1) 关于大数据,一看就懂,一懂就懵. 大数据的发展也有些年头了,如今正走在风口浪尖上,作为小白,我也来凑一份热闹. 大数据经过多年的发展,有着不同的 ...
- Luogu3731 HAOI2017新型城市化(二分图匹配+强连通分量)
将未建立贸易关系看成连一条边,那么这显然是个二分图.最大城市群即最大独立集,也即n-最大匹配.现在要求的就是删哪些边会使最大匹配减少,也即求哪些边一定在最大匹配中. 首先范围有点大,当然是跑个dini ...