题解 LNOI2014 LCA
题目:传送门
这道题根本不用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的更多相关文章
- 【BZOJ3626】[LNOI2014]LCA 离线+树链剖分+线段树
[BZOJ3626][LNOI2014]LCA Description 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1.设dep[i]表示点i的深度 ...
- BZOJ 3626: [LNOI2014]LCA [树链剖分 离线|主席树]
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2050 Solved: 817[Submit][Status ...
- bzoj 3626 [LNOI2014]LCA(离线处理+树链剖分,线段树)
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1272 Solved: 451[Submit][Status ...
- bzoj 3626: [LNOI2014]LCA 离线+树链剖分
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 426 Solved: 124[Submit][Status] ...
- BZOJ 3626: [LNOI2014]LCA( 树链剖分 + 离线 )
说多了都是泪啊...调了这么久.. 离线可以搞 , 树链剖分就OK了... -------------------------------------------------------------- ...
- 洛谷 P4211 [LNOI2014]LCA 解题报告
[LNOI2014]LCA 题意 给一个\(n(\le 50000)\)节点的有根树,询问\(l,r,z\),求\(\sum_{l\le i\le r}dep[lca(i,z)]\) 一直想启发式合并 ...
- P4211 [LNOI2014]LCA
P4211 [LNOI2014]LCA 链接 分析: 首先一种比较有趣的转化是,将所有点到1的路径上都+1,然后z到1的路径上的和,就是所有答案的deep的和. 对于多次询问,要么考虑有把询问离线,省 ...
- P4211 [LNOI2014]LCA LCT
P4211 [LNOI2014]LCA 链接 loj luogu 思路 多次询问\(\sum\limits_{l \leq i \leq r}dep[LCA(i,z)]\) 可以转化成l到r上的点到根 ...
- [BZOJ3626] [LNOI2014]LCA(树链剖分)
[BZOJ3626] [LNOI2014]LCA(树链剖分) 题面 给出一棵N个点的树,要求支持Q次询问,每次询问一个点z与编号为区间[l,r]内的点分别求最近公共祖先得到的最近公共祖先深度和.N, ...
随机推荐
- [置顶]
使用 maven 插件 maven-shade-plugin 对可执行 java 工程及其全部依赖 jar 进行打包
作者:chenzhou123520 出处:http://chenzhou123520.iteye.com/blog/1706242 使用 maven 插件 maven-shade-plugin 对 j ...
- 打包成ipa包
http://zengwu3915.blog.163.com/blog/static/27834897201362831449893/?suggestedreading&wumii Xcode ...
- poj 3420 Quad Tiling (状压dp+多米诺骨牌问题+矩阵快速幂)
还有这种操作?????? 直接用pre到now转移的方式构造一个矩阵就好了. 二进制长度为m,就构造一个长度为1 << m的矩阵 最后输出ans[(1 << m) - 1][( ...
- STM32 软件复位并模拟USB拔插
最近做了个USB跟上位机的通信,需要软件对MCU进行复位,复位后如果USB没有拔插,PC就不会重新枚举USB为了解决这个问题,我做了软件复位跟,软件模拟USB拔插. 这里我用的是HAL库的软件复位,复 ...
- tp框架报错 Namespace declaration statement has to be the very first statement in the script
Namespace declaration statement has to be the very first statement in the script tp框架报这个错误,错误行数就是nam ...
- 【codeforces 747E】Comments
[题目链接]:http://codeforces.com/problemset/problem/747/E [题意] 给你一个类似递归的结构; 让你把每一层的字符串按照层,一层层地输出出来; 并输出层 ...
- tddl
淘宝根据自己的业务特点开发了TDDL(Taobao Distributed Data Layer 外号:头都大了 ?_Ob)框架,主要解决了分库分表对应用的透明化以及异构数据库之间的数据复制,它是一个 ...
- 一个ibatis映射文件的例子(包含增删改单查,多查)
<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE mapper PUBLIC "-/ ...
- 解决Myeclipse在调试(debug)时无法显示变量值问题
解决Myeclipse在调试(debug)时无法显示变量值问题 突然发现myeclipse在调试时当鼠标放在变量上面时无法显示变量值了 ctrl+shift+D居然提示cannot be resolv ...
- bzoj2748: [HAOI2012]音量调节(背包)
2748: [HAOI2012]音量调节 题目:传送门 题解: sb省选题..呵呵一眼背包: f[i][j]表示第i时刻能否为音量j 代码: #include<cstdio> #inclu ...