题目:传送门


这道题根本不用lca,也没有部分分。。。

考虑求两个点xy的lca的深度。

我们将x到树根所有点的值都加1,然后查询y到根的和,其实就是lca的深度。

所以本题离线一下上树剖乱搞就可以了。


AC代码如下:
718ms 17348Kib

 #include <iostream>
#include <cstdio>
#include <vector>
#include <algorithm> using namespace std; namespace StandardIO { template<typename T> inline void read (T &x) {
x=;T f=;char c=getchar();
for (; c<''||c>''; c=getchar()) if (c=='-') f=-;
for (; c>=''&&c<=''; c=getchar()) x=x*+c-'';
x*=f;
}
template<typename T> inline void write (T x) {
if (x<) putchar('-'),x=-x;
if (x>=) write(x/);
putchar(x%+'');
} } using namespace StandardIO; namespace Solve { const int MOD=;
const int N=; struct Tree {
int tree[N*],tag[N*];
void pushdown (int pos,int left,int right) {
if (tag[pos]) {
int mid=(left+right)/;
tree[pos*]+=(mid-left+)*tag[pos],tree[pos*]%=MOD;
tree[pos*+]+=(right-mid)*tag[pos],tree[pos*+]%=MOD;
tag[pos*]+=tag[pos],tag[pos*+]+=tag[pos],tag[pos*]%=MOD,tag[pos*+]%=MOD;
tag[pos]=;
}
}
void pushup (int pos) {
tree[pos]=tree[pos*]+tree[pos*+],tree[pos]%=MOD;
}
void update (int pos,int left,int right,int L,int R,int add) {
if (L<=left&&right<=R) {
tree[pos]+=add*(right-left+),tree[pos]%=MOD;
tag[pos]+=add,tag[pos]%=MOD;
return;
}
pushdown(pos,left,right);
int mid=(left+right)/;
if (L<=mid) update(pos*,left,mid,L,R,add);
if (R>mid) update(pos*+,mid+,right,L,R,add);
pushup(pos);
}
int query (int pos,int left,int right,int L,int R) {
if (L<=left&&right<=R) return tree[pos];
pushdown(pos,left,right);
int mid=(left+right)/;
int ans=;
if (L<=mid) ans+=query(pos*,left,mid,L,R),ans%=MOD;
if (R>mid) ans+=query(pos*+,mid+,right,L,R),ans%=MOD;
return ans;
}
} ljz;
int n,q;
vector<int>M[N];
int dep[N],siz[N],fa[N],son[N];
int ind[N],cnt;
int top[N]; void dfs1 (int now,int father) {
dep[now]=dep[father]+,fa[now]=father,siz[now]=;
for (register vector<int>::iterator i=M[now].begin(); i!=M[now].end(); i++) {
if(*i==father) continue;
dfs1(*i,now);
siz[now]+=siz[*i];
if (siz[*i]>siz[son[now]]) son[now]=*i;
}
}
void dfs2(int now,int tp){
top[now]=tp,ind[now]=++cnt;
if (son[now]) dfs2(son[now],tp);
for (register vector<int>::iterator i=M[now].begin(); i!=M[now].end(); i++) {
if (*i==fa[now]||*i==son[now]) continue;
dfs2(*i,*i);
}
}
void upd (int x,int add){
while (x) {
ljz.update(,,n,ind[top[x]],ind[x],add);
x=fa[top[x]];
}
}
int que (int x) {
int ans=;
while (x) {
ans+=ljz.query(,,n,ind[top[x]],ind[x]),ans%=MOD;
x=fa[top[x]];
}
return ans;
}
int A[N];
struct Q{
int val,ind;
Q () {val=ind=;}
Q (int a,int b) :val(a),ind(b) {}
friend bool operator < (Q a,Q b) {
return a.val<b.val;
}
} s1[N],s2[N];
int a1=,a2=;
int Ans[N]; inline void solve () {
read(n),read(q);
for (register int i=; i<=n; i++) {
int a;
read(a);
M[a+].push_back(i);
}
dfs1(,);
dfs2(,);
for (register int i=; i<=q; i++) {
int a,b,c;
read(a),read(b),read(c);
a++,b++,c++;
A[i]=c;
s1[i]=Q(a-,i),s2[i]=Q(b,i);
}
sort(s1+,s1+q+);
sort(s2+,s2+q+);
while (s1[a1].val==) a1++;
for (register int i=; i<=n; i++) {
upd(i,);
while (s1[a1].val==i) {
Ans[s1[a1].ind]-=que(A[s1[a1].ind])-MOD,Ans[s1[a1].ind]%=MOD;
a1++;
}
while (s2[a2].val==i) {
Ans[s2[a2].ind]+=que(A[s2[a2].ind]),Ans[s2[a2].ind]%=MOD;
a2++;
}
}
for (register int i=; i<=q; i++) {
write(Ans[i]),putchar('\n');
}
} } using namespace Solve; int main () {
// freopen(".in","r",stdin);
// freopen(".out","w",stdout);
solve();
}

题解 LNOI2014 LCA的更多相关文章

  1. 【BZOJ3626】[LNOI2014]LCA 离线+树链剖分+线段树

    [BZOJ3626][LNOI2014]LCA Description 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1.设dep[i]表示点i的深度 ...

  2. BZOJ 3626: [LNOI2014]LCA [树链剖分 离线|主席树]

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2050  Solved: 817[Submit][Status ...

  3. bzoj 3626 [LNOI2014]LCA(离线处理+树链剖分,线段树)

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1272  Solved: 451[Submit][Status ...

  4. bzoj 3626: [LNOI2014]LCA 离线+树链剖分

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 426  Solved: 124[Submit][Status] ...

  5. BZOJ 3626: [LNOI2014]LCA( 树链剖分 + 离线 )

    说多了都是泪啊...调了这么久.. 离线可以搞 , 树链剖分就OK了... -------------------------------------------------------------- ...

  6. 洛谷 P4211 [LNOI2014]LCA 解题报告

    [LNOI2014]LCA 题意 给一个\(n(\le 50000)\)节点的有根树,询问\(l,r,z\),求\(\sum_{l\le i\le r}dep[lca(i,z)]\) 一直想启发式合并 ...

  7. P4211 [LNOI2014]LCA

    P4211 [LNOI2014]LCA 链接 分析: 首先一种比较有趣的转化是,将所有点到1的路径上都+1,然后z到1的路径上的和,就是所有答案的deep的和. 对于多次询问,要么考虑有把询问离线,省 ...

  8. P4211 [LNOI2014]LCA LCT

    P4211 [LNOI2014]LCA 链接 loj luogu 思路 多次询问\(\sum\limits_{l \leq i \leq r}dep[LCA(i,z)]\) 可以转化成l到r上的点到根 ...

  9. [BZOJ3626] [LNOI2014]LCA(树链剖分)

    [BZOJ3626] [LNOI2014]LCA(树链剖分) 题面 给出一棵N个点的树,要求支持Q次询问,每次询问一个点z与编号为区间[l,r]内的点分别求最近公共祖先得到的最近公共祖先深度和.N, ...

随机推荐

  1. [Vue warn]: Invalid prop: custom validator check failed for prop "type".

    遇到错误如下, [Vue warn]: Invalid prop: custom validator check failed for prop "type". found in ...

  2. [中文] 以太坊(Ethereum )白皮书

    以太坊(Ethereum ):下一代智能合约和去中心化应用平台 翻译|巨蟹 .少平 译者注|中文读者可以到以太坊爱好者社区(www.ethfans.org)获取最新的以太坊信息. 当中本聪在2009年 ...

  3. linux中一次创建多个目录

    linux中创建目录当然使用命令工具mkdir == (make directory),联想记忆法能让你记得牢固. 如果你要创建几个目录,例如:dir1目录,dir2目录,dir3目录可以这样 mkd ...

  4. JS数组中的indexOf方法

    前言 这两天在家中帮朋友做项目,项目中使用了数组的indexOf 方法,找到了一篇文章,感觉非常不错,顺便整理下以防链接丢失. 相信说到 indexOf 大家并不陌生,判断字符串是否包涵子字符串时特别 ...

  5. VUE:条件渲染和列表渲染

    条件渲染 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <titl ...

  6. (转)C++引用

    前言:引用是C++一个很重要的特性,最近看了很多有关引用的资料和博客,故在此对引用的相关知识进行总结 一.什么是引用 引用,顾名思义是某一个变量或对象的别名,对引用的操作与对其所绑定的变量或对象的操作 ...

  7. mysql字符集修改(ubuntu)

    1.关闭mysql服务 /etc/init.d/mysql start|stop 2.在/etc/mysql/my.cnf,添加下列信息 [client] default-character-set= ...

  8. Session小案例-----简单购物车的使用

    Session小案例-----简单购物车的使用 同上篇一样,这里的处理请求和页面显示相同用的都是servlet. 功能实现例如以下: 1,显示站点的全部商品 2.用户点击购买后,可以记住用户选择的商品 ...

  9. 数据仓库工具:Hive

    转载请标明出处: http://blog.csdn.net/zwto1/article/details/46430823: 本文出自:[明月的博客] 为什么要选择Hive 基于Hadoop的大数据的计 ...

  10. 小记 SqlHelper

    using System;using System.Collections.Generic;using System.Data;using System.Linq;using System.Web;u ...