【模板】树链剖分(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 这也是个模 ...
随机推荐
- 为git伸冤(使用git中可能会踩的一些坑)
窦娥被冤,六月飞霜.今天我两次遇到git被冤枉的情况,要是其它人也遇到这种情况导致git使用量缩小,那也要六月飞霜了. git第一次被冤枉是同事以为git的版本管理有问题.事情是这样的,同事提交新版本 ...
- python函数知识二 动态参数、函数的注释、名称空间、函数的嵌套、global,nonlocal
6.函数的动态参数 *args,**kwargs:能接受动态的位置参数和动态的关键字参数 *args -- tuple *kwargs -- dict 动态参数优先级:位置参数 > 动态位置参数 ...
- 2019牛客多校第一场A-Equivalent Prefixes
Equivalent Prefixes 传送门 解题思路 先用单调栈求出两个序列中每一个数左边第一个小于自己的数的下标, 存入a[], b[].然后按照1~n的顺序循环,比较 a[i]和b[i]是否相 ...
- mysql_fetch_assoc与mysql_fetch_array的区别
mysql_fetch_assoc与mysql_fetch_array的区别? 1. mysql_fetch_assoc : mysql_fetch_assoc() 函数从结果集中取得一行作为关联数组 ...
- Salesforce Admin篇(三) Delegated Administrator
项目中,我们可能会遇见以下的场景. 1. HR 经理针对申请者和工作相关的表的app会经常需要修改布局查看需要的页面的字段: 2. 开发者将record type对应的picklist values等 ...
- [WPF自定义控件库]自定义Expander
1. 前言 上一篇文章介绍了使用Resizer实现Expander简单的动画效果,运行效果也还好,不过只有展开/折叠而缺少了淡入/淡出的动画(毕竟Resizer模仿Expander只是附带的功能).这 ...
- 高级MySQL
一.MySQL的架构介绍 1.高级MySQL MySQL内核 SQL优化 MySQL服务器的优化 各种参数常亮设定 查询语句优化 主从复制 软硬件升级 容灾备份 SQL编程 2.MySQL的Linux ...
- difflib python
difflib -帮助进行差异化比较 这个模块提供的类和方法用来进行差异化比较,它能够生成文本或者html格式的差异化比较结果,如果需要比较目录的不同,可以使用filecmp模块. 例子: # -*- ...
- 手动创建MySQL服务
1.复制一份MySQL服务文件,放入一个路径 2.清理data文件夹下文件,仅保留mysql等 3.修改my.ini,port,...dir等配置 4.管理员运行cmd,cd>bin:mysql ...
- PyCharm 配置 Git 教程
之前给大家介绍了 Git 安装及使用指南,今天再给大家介绍一下在 PyCharm 中使用 Git. 1 打开 File -> Settings -> Version Control -&g ...