题面

挺好的一道树剖模板;

首先要学会最模板的树剖;

然后这道题要注意几个细节:

初始化时,seg[0]=1,seg[root]=1,top[root]=root,rev[1]=root;

在线段树上进行操作时,要使用lazy标记;

对于一个以x为根的子树,它子树中所有的元素一定时在线段树上连续的区间,且以seg[x]开始,以seg[x]+size[x]-1结束;

然后写码的时候注意不要手残(比如说预处理时写成了dep[u]=dep[u]+1);

#include <bits/stdc++.h>
using namespace std;
int n,m,r,p;
int head[2000010],cnt;
class littlestar{
public:
int to;
int nxt;
void add(int u,int v){
to=v;
nxt=head[u];
head[u]=cnt;
}
}star[2000010];
int a[100010];
int f[100010],dep[100010],son[100010],seg[100010],rev[100010],size[100010],top[100010];
void dfs1(int u,int fa)
{
size[u]=1;
f[u]=fa;
dep[u]=dep[fa]+1;
for(int i=head[u];i;i=star[i].nxt){
int v=star[i].to;
if(v==fa) continue;
dfs1(v,u);
size[u]+=size[v];
if(size[v]>size[son[u]]) son[u]=v;
}
}
void dfs2(int u,int fa)
{
if(son[u]){
seg[son[u]]=++seg[0];
rev[seg[0]]=son[u];
top[son[u]]=top[u];
dfs2(son[u],u);
}
for(int i=head[u];i;i=star[i].nxt){
int v=star[i].to;
if(v==fa) continue;
if(!top[v]){
seg[v]=++seg[0];
rev[seg[0]]=v;
top[v]=v;
dfs2(v,u);
}
}
}
struct ss{
int sum;
int lazy;
}tree[1000010];
void build(int k,int l,int r)
{
if(l==r){
tree[k].sum=a[rev[l]]%p;
return;
}
int mid=(l+r)/2;
build(k<<1,l,mid);
build(k<<1|1,mid+1,r);
tree[k].sum=tree[k<<1].sum+tree[k<<1|1].sum;
}
void pre()
{
dfs1(r,0);
seg[0]=seg[r]=1;
top[r]=r;
rev[1]=r;
dfs2(r,0);
build(1,1,seg[0]);
}
void pushdown(int k,int l,int r)
{
int mid=(l+r)/2;
tree[k<<1].lazy=(tree[k<<1].lazy+tree[k].lazy)%p;
tree[k<<1].sum=(tree[k<<1].sum+tree[k].lazy*(mid-l+1))%p;
tree[k<<1|1].lazy=(tree[k<<1|1].lazy+tree[k].lazy)%p;
tree[k<<1|1].sum=(tree[k<<1|1].sum+tree[k].lazy*(r-mid))%p;
tree[k].lazy=0;
}
int query(int k,int l,int r,int x,int y)
{
if(r<x||l>y){
return 0;
}
if(l>=x&&r<=y){
return tree[k].sum%p;
}
int mid=(l+r)/2;
pushdown(k,l,r);
return (query(k<<1,l,mid,x,y)+query(k<<1|1,mid+1,r,x,y))%p;
}
void change(int k,int l,int r,int x,int y,int goal)
{
if(r<x||l>y) return;
if(l>=x&&r<=y){
tree[k].sum=(tree[k].sum+(r-l+1)*goal)%p;
tree[k].lazy=(tree[k].lazy+goal)%p;
return;
}
pushdown(k,l,r);
int mid=(l+r)/2;
change(k<<1,l,mid,x,y,goal);
change(k<<1|1,mid+1,r,x,y,goal);
tree[k].sum=(tree[k<<1].sum+tree[k<<1|1].sum)%p;
}
void changeroad(int x,int y,int z)
{
int fx=top[x],fy=top[y];
while(fx!=fy){
if(dep[fx]<dep[fy]) swap(fx,fy),swap(x,y);
change(1,1,seg[0],seg[fx],seg[x],z);
x=f[fx];
fx=top[x];
}
if(dep[x]>dep[y]) swap(x,y);
change(1,1,seg[0],seg[x],seg[y],z);
}
int queryroad(int x,int y)
{
long long ans=0;
int fx=top[x],fy=top[y];
while(fx!=fy){
if(dep[fx]<dep[fy]) swap(fx,fy),swap(x,y);
ans=(ans+query(1,1,seg[0],seg[fx],seg[x]))%p;
x=f[fx];
fx=top[x];
}
if(dep[y]<dep[x]) swap(x,y);
ans=(ans+query(1,1,seg[0],seg[x],seg[y]))%p;
return ans%p;
}
void changetree(int x,int goal)
{
change(1,1,seg[0],seg[x],seg[x]+size[x]-1,goal);
return;
}
long long querytree(int x)
{
long long res=0;
res=(res+query(1,1,seg[0],seg[x],seg[x]+size[x]-1))%p;
return res;
}
int main(){
cin>>n>>m>>r>>p;
for(int i=1;i<=n;i++){
scanf("%d",&a[i]);
}
for(int i=1;i<=n-1;i++){
int u,v;
scanf("%d%d",&u,&v);
star[++cnt].add(u,v);
star[++cnt].add(v,u);
}
pre();
for(int i=1;i<=m;i++){
int type;
scanf("%d",&type);
if(type==1){
int x,y,z;
scanf("%d%d%d",&x,&y,&z);
changeroad(x,y,z);
}
else if(type==2){
int x,y;
scanf("%d%d",&x,&y);
cout<<queryroad(x,y)%p<<endl;
}
else if(type==3){
int x,z;
scanf("%d%d",&x,&z);
changetree(x,z);
}
else{
int x;
scanf("%d",&x);
cout<<querytree(x)%p<<endl;
}
}
}
/*
5 5 2 30000
7 3 7 8 0
1 2
1 5
3 1
4 1
3 4 2
3 2 2
4 5
1 5 1 3
2 1 3
*/

洛谷 P3384树链剖分 题解的更多相关文章

  1. 洛谷P3384 树链剖分

    如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式: 2 x ...

  2. 洛谷 P3384 树链剖分(模板题)

    题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式 ...

  3. 洛谷 [P3384] 树链剖分 模版

    支持各种数据结构上树,注意取膜. #include <iostream> #include <cstring> #include <algorithm> #incl ...

  4. 【算法学习】【洛谷】树链剖分 & P3384 【模板】树链剖分 P2146 软件包管理器

    刚学的好玩算法,AC2题,非常开心. 其实很早就有教过,以前以为很难就没有学,现在发现其实很简单也很有用. 更重要的是我很好调试,两题都是几乎一遍过的. 介绍树链剖分前,先确保已经学会以下基本技巧: ...

  5. 【树链剖分】洛谷P3379 树链剖分求LCA

    题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下来N-1行每 ...

  6. 洛谷P2146 树链剖分

    题意 思路:直接树链剖分,用线段树维护即可,算是树剖的经典题目吧. 代码: #include <bits/stdc++.h> #define ls(x) (x << 1) #d ...

  7. 洛谷p3384【模板】树链剖分题解

    洛谷p3384 [模板]树链剖分错误记录 首先感谢\(lfd\)在课上调了出来\(Orz\) \(1\).以后少写全局变量 \(2\).线段树递归的时候最好把左右区间一起传 \(3\).写\(dfs\ ...

  8. P3384 【模板】树链剖分 题解&&树链剖分详解

    题外话: 一道至今为止做题时间最长的题: begin at 8.30A.M 然后求助_yjk dalao后 最后一次搞取模: awsl. 正解开始: 题目链接. 树链剖分,指的是将一棵树通过两次遍历后 ...

  9. 【树链剖分】洛谷P3384树剖模板

    题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式 ...

随机推荐

  1. SDOI2019R2游记

    Day 0 上午到了济南,住在了山下.下午颓颓颓,zhy在玩炉石,我在...打元气!我的机器人终于不掉HP通关了呢,送的皮肤好好看啊. Day 1 到考场后,打开题面,一看第一题似乎很可做啊,好像可以 ...

  2. 2、dubbo基础知识

    1.简介 2.dubbo架构 3.dubbo环境搭建 注意:cmd命令都是在bin目录的地址栏直接输入 xxx.cmd 4.配置dubbo-admin 步骤一: 步骤二: 步骤三: 步骤四: 步骤五: ...

  3. 51nod 1165 整边直角三角形的数量(两种解法)

    链接:http://www.51nod.com/Challenge/Problem.html#!#problemId=1165 直角三角形,三条边的长度都是整数.给出周长N,求符合条件的三角形数量. ...

  4. LeetCode---Sort && Segment Tree && Greedy

    307. Range Sum Query - Mutable 思路:利用线段树,注意数据结构的设计以及建树过程利用线段树,注意数据结构的设计以及建树过程 public class NumArray { ...

  5. leetcode题目10.正则表达式匹配(困难)

    题目描述: 给你一个字符串 s 和一个字符规律 p,请你来实现一个支持 '.' 和 '*' 的正则表达式匹配. '.' 匹配任意单个字符'*' 匹配零个或多个前面的那一个元素所谓匹配,是要涵盖 整个  ...

  6. windos批处理启动redis与哨兵

    为各个启动单独建立脚本后用总的bat调用 创建脚本,redis6379.bat脚本内容:@echo offtitle redis-serverset ENV_HOME6379="G:\Red ...

  7. OpenCV学习笔记(12)——OpenCV中的轮廓

    什么是轮廓 找轮廓.绘制轮廓等 1.什么是轮廓 轮廓可看做将连续的点(连着边界)连在一起的曲线,具有相同的颜色和灰度.轮廓在形态分析和物体的检测和识别中很有用. 为了更加准确,要使用二值化图像.在寻找 ...

  8. np.array()与np.asarray()区别

    1. 数据源a是数组ndarray时,array仍然会copy出一个副本,占用新的内存,但asarray不会.也就是说改变a的值,b不会. # 数据源a是列表时,两者没区别 a=[[1,2,3],[4 ...

  9. SDK location not found. Define location with sdk.dir in the local.properties file or with an ANDROID_HOME environment variable.

    问题描述: 已经安装了android-sdk 和gradle环境,并配置了环境变量,如下所示: android环境 root@wangju-HP--G4:/home/wangju/Desktop/5i ...

  10. MySQL操作符与相关函数

    union(联合)union使用是关联两张表或者两个查询所查出来的数据,联合成一张表但不会出现重复数据,显示的字段必须匹配列数. select s3.id cid,s1.cityName provin ...