【模板】树链剖分(Luogu P3384)
众所周知
树链剖分是个好东西QWQ
也是一个代码量破百的算法
基本定义

具体步骤
#include<bits/stdc++.h>
using namespace std;
template <typename T> inline void read(T &x)
{
x=0;int f=1;char c=getchar();
for(;c>'9'||c<'0';c=getchar()) if(c=='-') f=-1;
for(;c>='0'&&c<='9';c=getchar()) x=(x<<1)+(x<<3)+(c^48);
x*=f;
}
template <typename T> inline void print(T x)
{
if(x<0) putchar('-'),x=-x;
if(x>9) print(x/10);
putchar(x%10+48);
}
int tree[1010001],lazy[1010101];
struct ss{
int v;
int next;
int u;
};
ss s1[1010001];
int head[1101001],cnt;
int w[1010101],wt[1010101];
int n,m,r,mod;
int res=0;
int son[1010101],id[1010101],fa[1010101],tot,deep[1010100],size[1010101],top[1010010];
//--------------------------------------
inline void build(int rt,int l,int r)
{
if(l==r)
{
tree[rt]=wt[l];
tree[rt]%=mod;
return;
}
int mid=(l+r)>>1;
build(rt<<1,l,mid);
build(rt<<1|1,mid+1,r);
tree[rt]=(tree[rt<<1]+tree[rt<<1|1])%mod;
}
inline void pushdown(int rt,int len)
{
lazy[rt<<1]+=lazy[rt];
lazy[rt<<1|1]+=lazy[rt];
tree[rt<<1]+=lazy[rt]*(len-(len>>1));
tree[rt<<1|1]+=lazy[rt]*(len>>1);
tree[rt<<1]%=mod;
tree[rt<<1|1]%=mod;
lazy[rt]=0;
}
inline void query(int rt,int l,int r,int L,int R)
{
int mid=(l+r)>>1;
if(L<=l&&r<=R)
{
res+=tree[rt],res%=mod;
return;
}
else
{
if(lazy[rt])
pushdown(rt,(r-l+1));
if(L<=mid)
query(rt<<1,l,mid,L,R);
if(R>mid)
query(rt<<1|1,mid+1,r,L,R);
}
}
inline void update(int rt,int l,int r,int L,int R,int k)
{
int mid=(l+r)>>1;
if(L<=l&&r<=R)
lazy[rt]+=k,tree[rt]+=k*(r-l+1);
else
{
if(lazy[rt])
pushdown(rt,r-l+1);
if(L<=mid)
update(rt<<1,l,mid,L,R,k);
if(R>mid)
update(rt<<1|1,mid+1,r,L,R,k);
tree[rt]=(tree[rt<<1]+tree[rt<<1|1])%mod;
}
}
//--------------------------------------
inline void abb(int x,int y)
{
cnt++;
s1[cnt].u=x;
s1[cnt].v=y;
s1[cnt].next=head[x];
head[x]=cnt;
}
inline void DFS(int x,int f,int dep)
{
deep[x]=dep;
fa[x]=f;
size[x]=1;
int MAXX=-1;
for(register int i=head[x];i;i=s1[i].next)
{
int v=s1[i].v;
if(v==f)
continue;
DFS(v,x,dep+1);
size[x]+=size[v];
if(size[v]>MAXX)
son[x]=v,MAXX=size[v];
}
}
inline void DFS_(int x,int TOP)
{
id[x]=++tot;
wt[tot]=w[x];
top[x]=TOP;
if(!son[x])
return;
DFS_(son[x],TOP);
for(register int i=head[x];i;i=s1[i].next)
{
int v=s1[i].v;
if(v==fa[x]||v==son[x])
continue;
DFS_(v,v);
}
}
//--------------------------------------------
inline int qSon(int x)
{
res=0;
query(1,1,n,id[x],id[x]+size[x]-1);
return res;
}
inline void updSon(int x,int k)
{
update(1,1,n,id[x],id[x]+size[x]-1,k);
}
inline void updRange(int x,int y,int k)
{
k%=mod;
while(top[x]!=top[y])
{
if(deep[top[x]]<deep[top[y]])
swap(x,y);
update(1,1,n,id[top[x]],id[x],k);
x=fa[top[x]];
}
if(deep[x]>deep[y])
swap(x,y);
update(1,1,n,id[x],id[y],k);
}
inline int qRange(int x,int y)
{
int ans=0;
while(top[x]!=top[y])
{
if(deep[top[x]]<deep[top[y]])
swap(x,y);
res=0,query(1,1,n,id[top[x]],id[x]),ans+=res,ans%=mod,x=fa[top[x]];
}
if(deep[x]>deep[y])
swap(x,y);
res=0,query(1,1,n,id[x],id[y]),ans+=res;
return ans%mod;
}
int main()
{
read(n),read(m),read(r),read(mod);
for(register int i=1;i<=n;++i)
read(w[i]);
for(register int i=1,a,b;i<n;++i)
read(a),read(b),abb(a,b),abb(b,a);
DFS(r,0,1);
DFS_(r,r);
build(1,1,n);
while(m--)
{
int a,b,c,q;
read(q);
if(q==1)
read(a),read(b),read(c),updRange(a,b,c);
else if(q==2)
read(a),read(b),print(qRange(a,b)),putchar('\n');
else if(q==3)
read(a),read(b),updSon(a,b);
else
read(a),print(qSon(a)),putchar('\n');
}
return 0;
}
【模板】树链剖分(Luogu P3384)的更多相关文章
- [luogu P3384] [模板]树链剖分
[luogu P3384] [模板]树链剖分 题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点 ...
- luoguP3384 [模板]树链剖分
luogu P3384 [模板]树链剖分 题目 #include<iostream> #include<cstdlib> #include<cstdio> #inc ...
- 树链剖分 - Luogu 3384【模板】树链剖分
[模板]树链剖分 题目描述 已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操 ...
- [洛谷P3384] [模板] 树链剖分
题目传送门 显然是一道模板题. 然而索引出现了错误,狂wa不止. 感谢神犇Dr_J指正.%%%orz. 建线段树的时候,第44行. 把sum[p]=bv[pos[l]]%mod;打成了sum[p]=b ...
- P3384 [模板] 树链剖分
#include <bits/stdc++.h> using namespace std; typedef long long ll; int n, m, rt, mod, cnt, to ...
- 模板 树链剖分BFS版本
//点和线段树都从1开始 //边使用vector vector<int> G[maxn]; ],num[maxn],iii[maxn],b[maxn],a[maxn],top[maxn], ...
- 『题解』洛谷P3384 【模板】树链剖分
Problem Portal Portal1: Luogu Description 如题,已知一棵包含\(N\)个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作\(1\): ...
- 树链剖分详解(洛谷模板 P3384)
洛谷·[模板]树链剖分 写在前面 首先,在学树链剖分之前最好先把 LCA.树形DP.DFS序 这三个知识点学了 emm还有必备的 链式前向星.线段树 也要先学了. 如果这三个知识点没掌握好的话,树链剖 ...
- Luogu P3384 【【模板】树链剖分】
转载请注明出处,部分内容引自banananana大神的博客 ~~别说你不知道什么是树~~╮(─▽─)╭(帮你百度一下) 先来回顾两个问题:1,将树从x到y结点最短路径上所有节点的值都加上z 这也是个模 ...
随机推荐
- JVM调优之探索CMS和G1的物理内存归还机制
前言: 公司有一个资产统计系统,使用频率很低,但是要求在使用时查询速度快,因此想到做一些缓存放在内存中,在长时间没有使用,持久化到磁盘中,并对垃圾进行回收,归还物理内存给操作系统,从而节省宝贵资源给其 ...
- 浅谈hosts文件
1.什么是hosts文件?这个文件在哪? hosts文件(域名解析文件)是将主机名映射到IP地址的一个纯文本文件,原始名称是HOSTS.TXT(IP,Internet Protocol,Interne ...
- random,time,sys,os
import random print(random.random()) #(0,1)大于0且小于1之间的小数 print(random.randint(1,3)) #大于等于1且小于等于3之间的整数 ...
- Java--UTF-8
UTF-8(8-bit Unicode Transformation Format)是一种针对Unicode的可变长度字符编码,又称万国码.由Ken Thompson于1992年创建.现在已经标准化为 ...
- 基于drone构建CI-CD系统
kubernetes集群三步安装 CI 概述 用一个可描述的配置定义整个工作流 程序员是很懒的动物,所以想各种办法解决重复劳动的问题,如果你的工作流中还在重复一些事,那么可能就得想想如何优化了 持续集 ...
- 移动端H5多终端适配解决方案
推荐三篇文章: 1.来自手淘团队的开源技术flexibal.js github地址:https://github.com/isHelenaChan/flexible 2.来自“大漠”(就职于淘宝,也是 ...
- Spring WebClient vs. RestTemplate
1. 简介 本教程中,我们将对比 Spring 的两种 Web 客户端实现 -- RestTemplate 和 Spring 5 中全新的 Reactive 替代方案 WebClient. 2. 阻塞 ...
- 【iOS】Ineligible Devices || “无法下载应用程序”
今天遇到了这个问题,Xcode 显示如图所示: 还有真机测试无法安装的问题,如图: 究其原因,都是 版本不匹配 的问题!在 Xcode 中的 PROJECT 和 TARGETS 设置下版本就行了,如下 ...
- 【SVN】eclipse 安装 SVN 插件
链接:eclipse中svn插件的安装 SVN 插件地址:http://subclipse.tigris.org/servlets/ProjectProcess;jsessionid=8EB28B11 ...
- 爬虫环境搭建及 scrapy 启动
创建虚拟环境 C:\Users\Toling>mkvirtualenv article 这个是普通的创建虚拟环境,但是实际开发中可能会使用python2或python3所以我们需要指定开发的环境 ...