支持各种数据结构上树,注意取膜.

#include <iostream>
#include <cstring>
#include <algorithm>
#include <cstdlib>
#include <cstdio>
#include <queue>
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
#define LL long long
using namespace std;
const int MAXN=100005;
int init(){
int rv=0,fh=1;
char c=getchar();
while(c<'0'||c>'9'){
if(c=='-') fh=-1;
c=getchar();
}
while(c>='0'&&c<='9'){
rv=(rv<<1)+(rv<<3)+c-'0';
c=getchar();
}
return fh*rv;
}
int ind,head[MAXN],dep[MAXN],son[MAXN],nume,n,m,MOD,root,id[MAXN],siz[MAXN],wei[MAXN],wein[MAXN],top[MAXN],fa[MAXN];
struct edge{
int to,nxt;
}e[MAXN<<1];
void adde(int from,int to){
e[++nume].to=to;
e[nume].nxt=head[from];
head[from]=nume;
}
void dfs1(int u,int rt){
dep[u]=dep[rt]+1;
fa[u]=rt;
siz[u]=1;
int ma=0;
for(int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
if(v==rt) continue;
dfs1(v,u);
siz[u]+=siz[v];
if(siz[v]>ma){
ma=siz[v];
son[u]=v;
}
}
}
void dfs2(int u,int topf){
id[u]=++ind;
top[u]=topf;
wein[ind]=wei[u];
if(!son[u]) return ;
dfs2(son[u],topf);
for(int i=head[u];i;i=e[i].nxt){
int v=e[i].to;
if(v==fa[u]||v==son[u]) continue;
dfs2(v,v);
}
}
struct SGT{
LL sum[MAXN<<2],tag[MAXN<<2];
void PushUp(int rt){
sum[rt]=(sum[rt<<1]+sum[rt<<1|1])%MOD;
}
void build(int l,int r,int rt){
if(l==r) {
sum[rt]=wein[l];
return;
}
int mid=(l+r)>>1;
if(l<=mid) build(lson);
if(mid<r) build(rson);
PushUp(rt);
}
void PushDown(int rt,int m){
if(tag[rt]){
(tag[rt<<1]+=tag[rt])%=MOD;
(tag[rt<<1|1]+=tag[rt])%=MOD;
(sum[rt<<1]+=tag[rt]*(m-(m>>1)))%=MOD;
(sum[rt<<1|1]+=tag[rt]*(m>>1))%=MOD;
tag[rt]=0;
}
}
void Update(int L,int R,int add,int l,int r,int rt){
if(L<=l&&r<=R){
(tag[rt]+=add)%=MOD;
(sum[rt]+=(LL)add*(r-l+1))%=MOD;
return;
}
int mid=(l+r)>>1;
PushDown(rt,r-l+1);
if(L<=mid) Update(L,R,add,lson);
if(mid<R) Update(L,R,add,rson);
PushUp(rt);
}
int query(int L,int R,int l,int r,int rt){
if(L<=l&&r<=R){
return sum[rt];
}
LL ret=0;
int mid=(l+r)>>1;
PushDown(rt,r-l+1);
if(L<=mid) ret+=query(L,R,lson);
if(mid<R) ret+=query(L,R,rson);
return ret%MOD;
}
}sgt;
void UpdateRange(int u,int v,int add){
add%=MOD;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]) swap(u,v);
sgt.Update(id[top[u]],id[u],add,1,n,1);
u=fa[top[u]];
}
if(dep[u]>dep[v]) swap(u,v);
sgt.Update(id[u],id[v],add,1,n,1);
}
int QueryRange(int u,int v){
int ans=0;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]) swap(u,v);
(ans+=sgt.query(id[top[u]],id[u],1,n,1))%=MOD;
u=fa[top[u]];
}
if(dep[u]>dep[v]) swap(u,v);
(ans+=sgt.query(id[u],id[v],1,n,1))%=MOD;
return ans%MOD;
}
int main(){
n=init();m=init();root=init();MOD=init();
for(int i=1;i<=n;i++) wei[i]=init();
for(int i=1;i<n;i++){
int u=init(),v=init();
adde(u,v);adde(v,u);
}
dep[root]=1;
dfs1(root,0);
dfs2(root,root);
sgt.build(1,n,1);
for(int i=1;i<=m;i++){
int t=init();
if(t==1){
int u=init(),v=init(),ttt=init();
UpdateRange(u,v,ttt);
}else if(t==2){
int u=init(),v=init();
printf("%d\n",QueryRange(u,v));
}else if(t==3){
int tt=init(),ttt=init();
sgt.Update(id[tt],id[tt]+siz[tt]-1,ttt%MOD,1,n,1);
}else {
int ttt=init();
printf("%d\n",sgt.query(id[ttt],id[ttt]+siz[ttt]-1,1,n,1));
}
}
return 0;
}

洛谷 [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树链剖分 题解

    题面 挺好的一道树剖模板: 首先要学会最模板的树剖: 然后这道题要注意几个细节: 初始化时,seg[0]=1,seg[root]=1,top[root]=root,rev[1]=root; 在线段树上 ...

  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树剖模板

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

  8. 洛谷树剖模板题 P3384 | 树链剖分

    原题链接 对于以u为根的子树,后代节点的dfn显然比他的dfn大,我们可以记录一下回溯到u的dfn,显然这两个dfn构成了一个连续区间,代表u及u的子树 剩下的就和树剖一样了 #include< ...

  9. P3384——树链剖分&&模板

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

随机推荐

  1. java web开发 高并发处理

    转自:http://blog.csdn.net/zhangzeyuaaa/article/details/44542161 java处理高并发高负载类网站中数据库的设计方法(java教程,java处理 ...

  2. ASP.NET CORE MVC 2.0 项目中引用第三方DLL报错的解决办法 - InvalidOperationException: Cannot find compilation library location for package

    目前在学习ASP.NET CORE MVC中,今天看到微软在ASP.NET CORE MVC 2.0中又恢复了允许开发人员引用第三方DLL程序集的功能,感到甚是高兴!于是我急忙写了个Demo想试试,我 ...

  3. POJ 2387 Til the Cows Come Home(模板——Dijkstra算法)

    题目连接: http://poj.org/problem?id=2387 Description Bessie is out in the field and wants to get back to ...

  4. HTTP协议简介

    一.简介 HTTP(HyperText Transfer Protocol, 超文本传输协议) 是访问互联网使用的核心通信协议,也是所有web应用程序使用的通信协议.消息模型:客户端发送请求消息,服务 ...

  5. 【编程技巧】EXTJS中Ext.grid.GridPanel配置项autoExpandColumn的使用方法

    autoExpandColumn的作用是自动伸展,占满剩余区域.一般使用在列比较少,并且大多数列都比较窄,有一列比较宽的情况下,当然什么时候使用,还是得按照实际情况确定. 使用的时候主要有三点要注意的 ...

  6. MySQL 查询最近几天的记录 最近7天的记录 本周内的记录

    本周内:select * from wap_content where week(created_at) = week(now) 查询一天:select * from table where to_d ...

  7. LindDotNetCore~Polly组件对微服务场景的价值

    回到目录 Polly是一个开源框架,在github上可以找到,被善友大哥收录,也是.App vNext的一员! App vNext:https://github.com/App-vNext GitHu ...

  8. mybatis实战教程三:mybatis和springmvc整合

    一.搭建maven环境,引入相关jar 参考demo 二.写web.xml <?xml version="1.0" encoding="UTF-8"?&g ...

  9. scrapy_创建_调试

    如何创建scrapy项目? 输入命令: scrapy startproject project_name 在当前目录下创建名字叫project_name的scrapy项目 命令格式:scrapy st ...

  10. Storm集群安装与部署

    准备 1.三台虚拟机 192.168.1.128 Nimbus 192.168.1.131 Supervisor 192.168.1.132 Supervisor 2.JDK1.8 3.Zookeep ...