[LNOI2014]LCA(树链剖分)
题目:给你一棵树,给你n个询问,每个询问要求输出$\sum_{i=l}^{r}depth(LCA(i,z))$
细看看其实没有想象的那么难
大体思路:
1、对于每个询问,考虑打差分变成$\sum_{i=0}^{r}depth(LCA(i,z))-\sum_{i=0}^{l-1}depth(LCA(i,z))$
2、那么这里就是一个思考点了
如何表示$\sum_{i=0}^{p}depth(LCA(i,z))$?
我也不知道
这种时候就该换一个角度思考了
啥是深度?
不就是从根到他有几个点嘛
对于两个点i,j
从i向根的路径上+1
那么$query(root,j)$就是要求的$dep(LCA(i,j))$了
到这里思路就差不多出来了
①把每个询问分成两个,打差分
②排序询问,逐个插入$add(i,root,1)$
③每个询问查出$query(z,root)$
好了答案出来了
#include<cstdio>
#include<algorithm>
using std::sort;
#define mo 201314
)%mo;}
int n,m;
struct sumireko
{
int to,ne;
}e[];
],ecnt;
void addline(int f,int t)
{
e[++ecnt].to=t;
e[ecnt].ne=he[f];
he[f]=ecnt;
}
],hop[],size[],dson[],llen[],dep[],idn[],idnar;
void dfs1(int x)
{
size[x]=;
;
for(int i=he[x],t;i;i=e[i].ne)
{
t=e[i].to;
dfs1(t);
size[x]+=size[t];
if(lin<size[t])
{
lin=size[t];
dson[x]=t;
}
}
}
void dfs2(int x,int top)
{
idn[x]=++idnar;
hop[x]=top;
llen[top]++;
if(!dson[x]) return;
dfs2(dson[x],top);
for(int i=he[x],t;i;i=e[i].ne)
{
t=e[i].to;
if(t==dson[x]) continue;
dfs2(t,t);
}
}
struct usami
{
int f,id,r,z;
void set(int u,int i,int o,int p){f=u,id=i,r=o,z=p;}
bool friend operator < (usami a,usami b)
{
return a.r<b.r;
}
}qqq[];
int qcnt;
];
struct segtree
{
],del[],size[];
void build(int px,int pl,int pr)
{
if(pl==pr)
{
size[px]=;
return;
}
;
build(px<<,pl,mid);
build(px<<|,mid+,pr);
size[px]=size[px<<]+size[px<<|];
}
void fuckup(int px,int pl,int pr)
{
if(pl==pr) return;
v[px]=v[px<<]+v[px<<|];
MOD(v[px]);
}
void fuckdown(int px,int pl,int pr)
{
if(pl==pr) return;
if(del[px])
{
v[px<<]+=del[px]*size[px<<];MOD(v[px<<]);
del[px<<]+=del[px];MOD(del[px<<]);
v[px<<|]+=del[px]*size[px<<|];MOD(v[px<<|]);
del[px<<|]+=del[px];MOD(del[px<<|]);
del[px]=;
}
}
void add(int px,int pl,int pr,int vin,int ql,int qr)
{
if(ql<=pl&&qr>=pr)
{
v[px]+=vin*size[px];
del[px]+=vin;
MOD(v[px]);
MOD(del[px]);
return;
}
fuckdown(px,pl,pr);
;
,pl,mid,vin,ql,qr);
|,mid+,pr,vin,ql,qr);
fuckup(px,pl,pr);
}
int query(int px,int pl,int pr,int ql,int qr)
{
if(ql<=pl&&qr>=pr) return v[px];
fuckdown(px,pl,pr);
;
;
,pl,mid,ql,qr),MOD(ret);
|,mid+,pr,ql,qr),MOD(ret);
return ret;
}
}tr;
void inp(int x)
{
])
{
tr.add(,,n,,idn[hop[x]],idn[x]);
x=fa[hop[x]];
}
tr.add(,,n,,idn[],idn[x]);
}
void qq(int x,int id,int f)
{
;
])
{
prt+=tr.query(,,n,idn[hop[x]],idn[x]);
MOD(prt);
x=fa[hop[x]];
}
prt+=tr.query(,,n,idn[],idn[x]);
MOD(prt);
ans[id]+=f*prt;
MOD(ans[id]);
}
int main()
{
scanf("%d%d",&n,&m);
;i<n;i++) scanf("%d",&fa[i]),addline(fa[i],i);
dep[]=;
dfs1();
dfs2(,);
tr.build(,,n);
,lin,rin,zin;i<=m;i++)
{
scanf("%d%d%d",&lin,&rin,&zin);
qqq[++qcnt].,i,lin-,zin);
qqq[++qcnt].,i,rin,zin);
}
sort(qqq+,qqq+qcnt+);
;
;i<=qcnt;i++)
{
while(tinar<qqq[i].r) {tinar++;inp(tinar);}
qq(qqq[i].z,qqq[i].id,qqq[i].f);
}
;i<=m;i++)
printf("%d\n",ans[i]);
;
}
精污代码慎入
[LNOI2014]LCA(树链剖分)的更多相关文章
- [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了... -------------------------------------------------------------- ...
- 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次询问,每次询 ...
- BZOJ 3626: [LNOI2014]LCA 树链剖分 线段树 离线
http://www.lydsy.com/JudgeOnline/problem.php?id=3626 LNOI的树链剖分题没有HAOI那么水,学到的东西还是很多的. 我如果现场写,很难想出来这种题 ...
- BZOJ 3626 [LNOI2014]LCA ——树链剖分
思路转化很巧妙. 首先把询问做差分. 然后发现加入一个点就把路径上的点都+1,询问的时候直接询问到根的路径和. 这样和原问题是等价的,然后树链剖分+线段树就可以做了. #include <map ...
- bzoj3626 [LNOI2014]LCA——树链剖分
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3626 思路很巧妙,把区间换成前缀和相减: 把 l ~ r 到根路径上的点的点权都+1,然后 ...
- 洛谷 P4211 [LNOI2014]LCA (树链剖分+离线)
题目:https://www.luogu.org/problemnew/solution/P4211 相当难的一道题,其思想难以用言语表达透彻. 对于每个查询,区间[L,R]中的每个点与z的lca肯定 ...
随机推荐
- 【HDU4706】Children's Day
http://acm.hdu.edu.cn/showproblem.php?pid=4706 水题,也不知道有没有spj // <4706.cpp> - 11/03/16 14:11:21 ...
- 【CSU 1803】2016
http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1803 Solution: 考虑两个数x,y乘积%2016=0 x×y≡0(MOD 2016) x= ...
- [翻译]NUnit--Getting Started(二)
Getting Started with NUnit 如果你打算开始学习,到下载页面选择一个NUnit版本.安装页面包含了安装说明. 开始NUnit阅读Quick Start页面.验证了一个C#银行应 ...
- Python不兼容问题
今天遇到了一个Python2与3不兼容的坑. ride是基于robot框架的python自动化ui,但它只支持python2,而我电脑环境只有python3,想跑别人基于ride编写的测试用例,折腾了 ...
- 【409】Linux 系统 Testrun
文件名:Testrun #!/bin/sh PROG=./puzzle case $1 in 1) T=Tests/bad* ;; 2) T=Tests/sol* ;; 3) T=Tests/unso ...
- 栗染-git命令搭建简单的个人的网页
本来一个很简单的东西被自己搞了很久 可能是对于一个小白来说第一次认识到github的魅力吧,以前只是听别人说过用github搭建网站,听起来很厉害的样子,一直也没有尝试过,突然兴起今天去网上找一些教程 ...
- robotframework - selenium 分层思路
前言: 对于每一条用例来说,调用“百度搜索”关键字,输入搜索内容,输入预期结果即可.不同关心用例是如何执行的.如果百度输入框的定位发生了变化,只用去修改“百度搜索”关键字即可,不用对每一条用例做任何修 ...
- Java 8 (4) Stream 流 - 使用
在本节中将介绍Stream API支持的许多操作,这些操作可以完成更复杂的数据查询,如筛选.切片.映射.查找.匹配和归约.还有一些特殊的流如:数值流.来自文件和数组等多种来源的流. 筛选和切片 1.用 ...
- Python(1)-第一天
PTVS下载地址:https://pytools.codeplex.com/releases/view/109707 Python下载地址:https://www.python.org/downloa ...
- 87. [NOIP2000] 乘积最大
★☆ 输入文件:cjzd.in 输出文件:cjzd.out 简单对比 时间限制:1 s 内存限制:128 MB 问题描述 今年是国际数学联盟确定的“2000——世界数学年”,又恰逢我国 ...