【BZOJ】3626 [LNOI2014]LCA
【算法】树链剖分+线段树(区间加值,区间求和)
【题解】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的更多相关文章
- 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: 2050 Solved: 817[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了... -------------------------------------------------------------- ...
- 【BZOJ】3052: [wc2013]糖果公园
http://www.lydsy.com/JudgeOnline/problem.php?id=3052 题意:n个带颜色的点(m种),q次询问,每次询问x到y的路径上sum{w[次数]*v[颜色]} ...
- 【BZOJ】3319: 黑白树
http://www.lydsy.com/JudgeOnline/problem.php?id=3319 题意:给一棵n节点的树(n<=1e6),m个操作(m<=1e6),每次操作有两种: ...
- 【BZOJ】4129: Haruna’s Breakfast 树分块+带修改莫队算法
[题意]给定n个节点的树,每个节点有一个数字ai,m次操作:修改一个节点的数字,或询问一条树链的数字集合的mex值.n,m<=5*10^4,0<=ai<=10^9. [算法]树分块+ ...
- 【BZOJ】3319: 黑白树(并查集+特殊的技巧/-树链剖分+线段树)
http://www.lydsy.com/JudgeOnline/problem.php?id=3319 以为是模板题就复习了下hld............................. 然后n ...
- 【BZOJ】1013: [JSOI2008]球形空间产生器sphere
[BZOJ]1013: [JSOI2008]球形空间产生器sphere 题意:给n+1个n维的点的坐标,要你求出一个到这n+1个点距离相等的点的坐标: 思路:高斯消元即第i个点和第i+1个点处理出一个 ...
随机推荐
- iOS开发GCD的简单使用
- (void)viewDidLoad { [super viewDidLoad]; // gcd 可以充分调用设备的 cpu 发挥最大性能,在 C 语言基础之上封装的 // dispatch_que ...
- 软工网络15团队作业4——Alpha阶段敏捷冲刺-1
各个成员在 Alpha 阶段认领的任务 成员 Alpha 阶段认领的任务 肖世松 编写界面设计代码 杨泽斌 服务器连接与配置 叶文柠 数据库连接与配置 谢庆圆 编写功能板块代码 林伟航 编写功能板块代 ...
- win7 php连接远程oracle
<?php /* 先下载oracle客户端 下载地址 http://www.oracle.com/technetwork/topics/winx64soft-089540.html 下载如下三个 ...
- Linux文件传输FTP详解
ftp命令用来设置文件系统相关功能.ftp服务器在网上较为常见,Linux ftp命令的功能是用命令的方式来控制在本地机和远程机之间传送文件,这里详细介绍Linux ftp命令的一些经常使用的命令,相 ...
- java文件操作(普通文件以及配置文件的读写操作)
转自:java文件操作(普通文件以及配置文件的读写操作) 读取普通文件 : /** * xiangqiao123欢迎你 如果对代码有疑问可以加qq群咨询:151648295 * * 读取MyFile文 ...
- Oracle中预定义角色有哪些?
1. CONNECT 2. RESOURCE 3. DBA 4. EXP_FULL_DATABASE 5. IMP_FULL_DATABASE 6. DELETE_CATALOG_ROLE 7. EX ...
- TScreen 类
TScreen表示应用程序运行时屏幕的状态. 类关系 TObject->TPersistent->TComponent TScreen引进具有表示下列各种情况的属性 什么窗体和数据模块已经 ...
- Windows 8.1 SecureBoot未正确配置的解决方法
使用联想Y510P,安装win8.1后破解 ,屏幕右下角老是显示 SecureBoot未正确配置的解决方法,以下是解决方案 步骤1:在机器重启至“Lenovo字样的屏幕”时,不停敲击“F2”键或“Fn ...
- BZOJ3139/BZOJ1306 HNOI2013比赛/CQOI2009循环赛(搜索)
搜索好难啊. 1.对于每个分数集合记忆化. 2.某人得分超过总分,剪枝. 3.某人之后全赢也无法达到总分,剪枝. 4.每有一场比赛分出胜负总分会多三分,而平局则会多两分.某人的分出胜负场次或平局场次超 ...
- 去除安卓手机select下拉框默认箭头
-webkit-appearance: listbox;