洛谷 P3384树链剖分 题解
挺好的一道树剖模板;
首先要学会最模板的树剖;
然后这道题要注意几个细节:
初始化时,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树链剖分 题解的更多相关文章
- 洛谷P3384 树链剖分
如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式: 2 x ...
- 洛谷 P3384 树链剖分(模板题)
题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式 ...
- 洛谷 [P3384] 树链剖分 模版
支持各种数据结构上树,注意取膜. #include <iostream> #include <cstring> #include <algorithm> #incl ...
- 【算法学习】【洛谷】树链剖分 & P3384 【模板】树链剖分 P2146 软件包管理器
刚学的好玩算法,AC2题,非常开心. 其实很早就有教过,以前以为很难就没有学,现在发现其实很简单也很有用. 更重要的是我很好调试,两题都是几乎一遍过的. 介绍树链剖分前,先确保已经学会以下基本技巧: ...
- 【树链剖分】洛谷P3379 树链剖分求LCA
题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下来N-1行每 ...
- 洛谷P2146 树链剖分
题意 思路:直接树链剖分,用线段树维护即可,算是树剖的经典题目吧. 代码: #include <bits/stdc++.h> #define ls(x) (x << 1) #d ...
- 洛谷p3384【模板】树链剖分题解
洛谷p3384 [模板]树链剖分错误记录 首先感谢\(lfd\)在课上调了出来\(Orz\) \(1\).以后少写全局变量 \(2\).线段树递归的时候最好把左右区间一起传 \(3\).写\(dfs\ ...
- P3384 【模板】树链剖分 题解&&树链剖分详解
题外话: 一道至今为止做题时间最长的题: begin at 8.30A.M 然后求助_yjk dalao后 最后一次搞取模: awsl. 正解开始: 题目链接. 树链剖分,指的是将一棵树通过两次遍历后 ...
- 【树链剖分】洛谷P3384树剖模板
题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式 ...
随机推荐
- [BJOI2015]树的同构 && 树哈希教程
题目链接 有根树的哈希 离散数学中对树哈希的描述在这里.大家可以看看. 判断有根树是否同构,可以考虑将有根树编码.而编码过程中,要求保留树形态的特征,同时忽略子树顺序的不同.先来看一看这个方法: 不妨 ...
- [CSP-S模拟测试]:C(三分+贪心)
题目传送门(内部题46) 输入格式 第一行$3$个整数$n,m,t$.第二行$n$个整数,表示$P_i$.接下来$m$行每行两个整数,表示$L_i,R_i$. 输出格式 一行一个整数表示答案. 样例 ...
- Android学习_7/27
一. 自定义控件 1. 引入布局 多个活动需要相同的布局时,使用引入布局的方式来实现代码复用. activity_main.xlm中加入<include la ...
- 设置PyCharm中选择文本的背景颜色和代码中和选中单词相同单词的背景颜色
1 设置选中单词的背景颜色 首先进入File->Setting->Editor->Color Scheme后复制一个存在的颜色主题作为自定义的颜色主题(默认的颜色主题是无法修改的,也 ...
- 黑马vue---10-11、Vue实现跑马灯效果
黑马vue---10-11.Vue实现跑马灯效果 一.总结 一句话总结: 1. 给 [浪起来] 按钮,绑定一个点击事件 v-on @ 2. 在按钮的事件处理函数中,写相关的业务逻辑代码:拿到 ...
- 黑马vue---1-7、vue杂记
黑马vue---1-7.vue杂记 一.总结 一句话总结: · 我最大的优势在于潜力,也就是孤独学习的能力.旁观者(l)看的比我清楚. · 那些游戏主播,比如英雄联盟主播,年复一年的玩一个游戏,一个英 ...
- STL priority_queue
priority_queue 优先队列(Priority Queues):顾名思义,一个有着优先级的队列.它是一种ADT,和队列的思想差不多—— 排队,数据结构中的队列是不能插队的,不能颠倒排队的顺序 ...
- 在已开启Chrome窗口上调试
代码 @Test void testNow() { /* First: Add the chrome.exe to the PATH. * Then: open the cmd and input t ...
- silverlight开发实例(Prism+MVVM+RIA)(二)--创建shell及用户登录
在上篇基本说清了本项目的基本框架,下面开始说下项目的加载和shell.开始之前在建立EF时出现了一个问题,我在数据库中建立了视图,而在EF导入视图时出现因无法匹配主键导致无法导入视图的问题,检查发现是 ...
- linux常用、常见错误
1.md5加密使用 oppnssl md5 加密字符串的方法 [root@lab3 ~]# openssl //在终端中输入openssl后回车. OpenSSL> md5 //输入md5后回车 ...