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

#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;
}

最新文章

  1. 触屏touchstart 与 click
  2. 树莓派2安装使用小米WIfi(360 小度 腾讯wifi)
  3. Remove openjdk in Ubuntu/Configure jdk and running adb in 64-bit Ubuntu
  4. HTTP、TCP、UDP以及SOCKET之间的区别/联系
  5. bnuoj 24251 Counting Pair
  6. 关于Hive的调优(本身,sql,mapreduce)
  7. fastboot 教程
  8. ERP存储过程的调用和树形菜单的加载(四)
  9. hadoop-集群管理(1)——配置文件
  10. HTML5 appcache
  11. mac下Android apk 破解流程
  12. HTTP Strict Transport Security实战详解
  13. HTML5网页点击分享到whatsapp
  14. URL和URI以及两者的区别和联系
  15. eclipse出现An internal error occurred during: &quot;Building workspace&quot;. Java heap space 错误
  16. TypeError: view must be a callable or a list/tuple in the case of include()
  17. git推送tag到远端服务器
  18. Jedis工具类(含分布式锁的调用和释放)
  19. BCB TLable控件透明背景属性
  20. 【FileZilla FTP Client】文件与服务器操作客户端

热门文章

  1. ACM中常见错误对应表
  2. 哈密顿绕行世界问题(dfs+记录路径)
  3. springboot注解使用说明
  4. block的修饰词为什么选用copy
  5. Centos6.7安装mysql 5.6简单教程
  6. Redis在Php项目中的实际应用场景
  7. 阿里大鱼 阿里云api
  8. eclipse怎么修改工作空间路径
  9. beetl 配置多视图解析器
  10. /dev/shm 引起的内存统计问题