题目:传送门


这道题根本不用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. CF482C Game with Strings (状压DP+期望DP)

    题目大意:甲和乙玩游戏,甲给出n(n<=50)个等长的字符串(len<=20),然后甲选出其中一个字符串,乙随机询问该字符串某一位的字符(不会重复询问一个位置),求乙能确定该串是哪个字符串 ...

  2. [CodeForces] 543B Destroying Roads

    脑洞+暴力. 因为边权是1,所以bfs一下,O(n^2)求任意两点间最短路,再枚举. ans最大是\(dis_{s1,t1}+dis_{s2,t2}\) 再考虑有公共边的情况,一定存在两个点 u, v ...

  3. Python 绘图与可视化 matplotlib 散点图、numpy模块的random()

    效果: 代码: def scatter_curve(): # plt.subplot(1,1,1) n=1024 X=np.random.normal(0,1,n) Y=np.random.norma ...

  4. Ubuntu下安装curl和corn

    Ubuntu下安装curl sudo apt install curl Ubuntu下安装cron apt-get install cron

  5. angular-resource

    上一篇中讲到使用$http同服务器进行通信,但是功能上比较简单,angularjs还提供了另外一个可选的服务$resource,使用它可以非常方便的同支持restful的服务单进行数据交互. 安装 n ...

  6. 【网络协议】TCP协议简单介绍

        本文仅仅是对TCP协议做个简要的介绍.     TCP协议,即传输控制协议.与UDP协议同处于传输层,相同使用相同的网络层,但TCP提供了一种可靠的.面向连接的传输数据服务,它会在两个使用TC ...

  7. java.util.concurrent.ExecutionException: org.apache.catalina.LifecycleException: Failed to start com

    错误如题. 原因:web.xml中的servlet映射<url-pattern> 配置错误 改动正确就可以. 我直接删除了,bug就攻克了. 另一个问题是 xxx.jar fail to ...

  8. POJ 3261 后缀数组+二分

    思路: 论文题- 二分+对后缀分组 这块一开始不用基数排序 会更快的(其实区别不大) //By SiriusRen #include <cstdio> #include <cstri ...

  9. Aspose office (Excel,Word,PPT),PDF 在线预览

    前文: 做个备份,拿的是试用版的 Aspose,功能见标题 代码: /// <summary> /// Aspose office (Excel,Word,PPT),PDF 在线预览 // ...

  10. 设备综合效率OEE

    设备综合效率OEE OEE(OverallEquipmentEffectiveness),即设备综合效率,也有资料表述为总体设备效率,其本质就是设备负荷时间内实际产量与理论产量的比值. TEEP(To ...