【算法】树链剖分+线段树(区间加值,区间求和)

【题解】http://hzwer.com/3891.html

中间不要取模不然相减会出错。

血的教训:线段树修改时标记下传+上传,查询时下传。如果修改时标记不下传,下面的结果就会覆盖上面的标记上传造成的影响。

读入后全部排序(离线处理)

链剖之后按顺序每个solve_insert(1,j),对于每次的z询问solve_sum(1,z)。

LCA其实就是两点到达根节点的路径的最近交点。

差分思想的运用:将区间差转为r-(l-1)。

#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
const int maxn=,MOD=;
int n,first[maxn],tot=,top[maxn],deep[maxn],pos[maxn],q,ansz[maxn],size[maxn],f[maxn],dfsnum=;
long long anss[maxn*];
struct edge{int from,v;}e[maxn*];
struct node{int l,r,delta,sum;}t[maxn*];
struct numbers{int num,ord;bool flag;}num[maxn*];
void insert(int u,int v)
{tot++;e[tot].v=v;e[tot].from=first[u];first[u]=tot;}
bool cmp(numbers a,numbers b)
{return a.num<b.num;}
void dfs1(int x,int fa)
{
size[x]=;
for(int i=first[x];i;i=e[i].from)
if(e[i].v!=fa)
{
int y=e[i].v;
f[y]=x;
deep[y]=deep[x]+;
dfs1(y,x);
size[x]+=size[y];
}
}
void dfs2(int x,int tp,int fa)
{
pos[x]=++dfsnum;
top[x]=tp;
int k=;
for(int i=first[x];i;i=e[i].from)
if(e[i].v!=fa&&size[e[i].v]>size[k])k=e[i].v;
if(k==)return;
dfs2(k,tp,x);
for(int i=first[x];i;i=e[i].from)
if(e[i].v!=fa&&e[i].v!=k)dfs2(e[i].v,e[i].v,x); }
void build(int k,int l,int r)
{
t[k].l=l;t[k].r=r;t[k].delta=;t[k].sum=;
if(l==r)return;
int mid=(l+r)>>;
build(k<<,l,mid);
build(k<<|,mid+,r);
t[k].sum=t[k<<].sum+t[k<<|].sum;
}
void add(int k,int l,int r)
{
int left=t[k].l,right=t[k].r;
if(l<=left&&right<=r)
{
t[k].delta++;
t[k].sum+=right-left+;
return;
}
if(t[k].delta)
{
t[k<<].delta+=t[k].delta;
t[k<<].sum+=(t[k<<].r-t[k<<].l+)*t[k].delta;
t[k<<|].delta+=t[k].delta;
t[k<<|].sum+=(t[k<<|].r-t[k<<|].l+)*t[k].delta;
t[k].delta=;
}
int mid=(left+right)>>;
if(l<=mid)add(k<<,l,r);
if(r>mid)add(k<<|,l,r);
t[k].sum=t[k<<].sum+t[k<<|].sum;
}
long long query(int k,int l,int r)
{
int left=t[k].l,right=t[k].r;
if(l<=left&&right<=r)return t[k].sum;
if(t[k].delta)
{
t[k<<].delta+=t[k].delta;
t[k<<].sum+=(t[k<<].r-t[k<<].l+)*t[k].delta;
t[k<<|].delta+=t[k].delta;
t[k<<|].sum+=(t[k<<|].r-t[k<<|].l+)*t[k].delta;
t[k].delta=;
}
int mid=(left+right)>>;
long long ans=;
if(l<=mid)ans=query(k<<,l,r);
if(r>mid)ans+=query(k<<|,l,r);
return ans;
}
void solve_ins(int x,int y)
{
while(top[x]!=top[y])
{
if(deep[top[x]]<deep[top[y]])swap(x,y);
add(,pos[top[x]],pos[x]);
x=f[top[x]];
}
if(pos[x]>pos[y])swap(x,y);
add(,pos[x],pos[y]);
}
long long solve_sum(int x,int y)
{
long long ans=;
while(top[x]!=top[y])
{
if(deep[top[x]]<deep[top[y]])swap(x,y);
ans+=query(,pos[top[x]],pos[x]);
x=f[top[x]];
}
if(pos[x]>pos[y])swap(x,y);
ans+=query(,pos[x],pos[y]);
return ans;
}
int main()
{
scanf("%d%d",&n,&q);
int u;
for(int i=;i<=n;i++)
{
scanf("%d",&u);
insert(u+,i);
insert(i,u+);
}
int ll,rr,zz;
for(int i=;i<=q;i++)
{
scanf("%d%d%d",&ll,&rr,&zz);ll++;rr++;zz++;
ansz[i]=zz;
num[i*-].num=ll-;num[i*-].ord=i;num[i*-].flag=;
num[i*].num=rr;num[i*].ord=i;num[i*].flag=;
}
sort(num+,num+q*+,cmp);
build(,,n);dfs1(,-);dfs2(,,-);
int now=;
memset(anss,,sizeof(anss));
for(int i=;i<=q*;i++)
{
if(num[i].num>now)
for(int j=now+;j<=num[i].num;j++)solve_ins(,j);
now=num[i].num;
if(num[i].flag)anss[num[i].ord]+=solve_sum(,ansz[num[i].ord]);
else anss[num[i].ord]-=solve_sum(,ansz[num[i].ord]);
} for(int i=;i<=q;i++)printf("%lld\n",anss[i]%MOD);
return ;
}

【BZOJ】3626 [LNOI2014]LCA的更多相关文章

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

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

  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: 426  Solved: 124[Submit][Status] ...

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

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

  5. 【BZOJ】3052: [wc2013]糖果公园

    http://www.lydsy.com/JudgeOnline/problem.php?id=3052 题意:n个带颜色的点(m种),q次询问,每次询问x到y的路径上sum{w[次数]*v[颜色]} ...

  6. 【BZOJ】3319: 黑白树

    http://www.lydsy.com/JudgeOnline/problem.php?id=3319 题意:给一棵n节点的树(n<=1e6),m个操作(m<=1e6),每次操作有两种: ...

  7. 【BZOJ】4129: Haruna’s Breakfast 树分块+带修改莫队算法

    [题意]给定n个节点的树,每个节点有一个数字ai,m次操作:修改一个节点的数字,或询问一条树链的数字集合的mex值.n,m<=5*10^4,0<=ai<=10^9. [算法]树分块+ ...

  8. 【BZOJ】3319: 黑白树(并查集+特殊的技巧/-树链剖分+线段树)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3319 以为是模板题就复习了下hld............................. 然后n ...

  9. 【BZOJ】1013: [JSOI2008]球形空间产生器sphere

    [BZOJ]1013: [JSOI2008]球形空间产生器sphere 题意:给n+1个n维的点的坐标,要你求出一个到这n+1个点距离相等的点的坐标: 思路:高斯消元即第i个点和第i+1个点处理出一个 ...

随机推荐

  1. java — JVM调优

    数据类型 Java虚拟机中,数据类型可以分为两类:基本类型和引用类型.基本类型的变量保存原始值,即:他代表的值就是数值本身:而引用类型的变量保存引用值.“引用值”代表了某个对象的引用,而不是对象本身, ...

  2. erlang+thrift配合开发

    I  think, thrift is a  tcp/ip based Client-Server architecture multi-languages supported RPC framewo ...

  3. OSG学习:移动/缩放/旋转模型

    移动和缩放以及旋转都是对矩阵进行操作,这些操作如果要叠加直接矩阵相乘就可以了. 下面的示例代码中,加入了四个bignathan,一个是默认加入在最中间,一个向上移2单位,一个是向下移2单位且缩放0.5 ...

  4. 3dContactPointAnnotationTool开发日志(二一)

      今天完成了修改按钮颜色,添加smpl模型到工具,以及可以显示物体子物体对应选项卡的功能.把之前的meshRenderer+meshFilter都改成了skinnedMeshRenderer,因为s ...

  5. mysql向上递归&向下递归

    工作记录 向上递归函数test: BEGIN ); ); SET sTemp = '$'; SET sTempChd =cast(rid as CHAR); WHILE sTempChd is not ...

  6. [剑指Offer] 55.链表中环的入口结点

    题目描述 一个链表中包含环,请找出该链表的环的入口结点. [思路]根据set集合的不重复,遍历链表时遇到的第一个重复结点就是环的入口结点. /* struct ListNode { int val; ...

  7. Redis Cluster实现原理

    一.Redis Cluster主要特性和设计     集群目标 1)高性能和线性扩展,最大可以支撑到1000个节点:Cluster架构中无Proxy层,Master与slave之间使用异步replic ...

  8. BZOJ 1407 Savage(拓展欧几里得)

    这题的时间复杂度真玄学... O(m*n^2).1e8也能过啊... 首先题目保证m<=1e6. 这启发我们枚举或者二分答案? 但是答案不满足单调性,考虑从小到大枚举m. 对于每一个m,枚举两个 ...

  9. 【题解】CF#960 H-Santa's Gift

    好久没有写过数据结构题目了,果然还是太不自信.实际上就是要求统计一个式子: \(\sum (c[k]*p[k] - C)^{2}\) 拆开,分别统计和与平方和 \(co[k] * \sum p[k]^ ...

  10. 洛谷 P2747 [USACO5.4]周游加拿大Canada Tour 解题报告

    P2747 [USACO5.4]周游加拿大Canada Tour 题目描述 你赢得了一场航空公司举办的比赛,奖品是一张加拿大环游机票.旅行在这家航空公司开放的最西边的城市开始,然后一直自西向东旅行,直 ...