原题链接

对于以u为根的子树,后代节点的dfn显然比他的dfn大,我们可以记录一下回溯到u的dfn,显然这两个dfn构成了一个连续区间,代表u及u的子树

剩下的就和树剖一样了

 #include<cstdio>
#include<algorithm>
#include<cstring>
#define N 100010
typedef long long ll;
using namespace std;
int ecnt,head[N],son[N],fa[N],sz[N],n,m,r,deep[N],pos[N],indx[N],tot,op,back[N],top[N];
ll val[N],P;
struct adj
{
int nxt,v;
}e[*N];
struct node
{
int l,r;
ll sum,lz;
}t[*N];
void add(int u,int v)
{
e[++ecnt].v=v;
e[ecnt].nxt=head[u];
head[u]=ecnt;
e[++ecnt].v=u;
e[ecnt].nxt=head[v];
head[v]=ecnt;
}
void dfs1(int x,int father,int dep)
{
deep[x]=dep,fa[x]=father,sz[x]=;
for (int i=head[x];i;i=e[i].nxt)
{
int v=e[i].v;
if (fa[x]==v) continue;
dfs1(v,x,dep+);
sz[x]+=sz[v];
if (sz[v]>sz[son[x]]) son[x]=v;
}
}
void dfs2(int x,int TOP)
{
top[x]=TOP;
pos[x]=++tot;
indx[tot]=x;
if (son[x]) dfs2(son[x],TOP);
for (int i=head[x];i;i=e[i].nxt)
{
int v=e[i].v;
if (v==fa[x] || v==son[x]) continue;
dfs2(v,v);
}
back[x]=tot;
}
void pushdown(int p)
{
if (t[p].l==t[p].r || !t[p].lz) return;
int w=t[p].lz;
t[p<<].sum=(t[p<<].sum+w*(t[p<<].r-t[p].l+)%P)%P;
t[p<<|].sum=(t[p<<|].sum+w*(t[p<<|].r-t[p<<|].l+)%P)%P;
t[p<<].lz+=w;
t[p<<].lz%=P;
t[p<<|].lz+=w;
t[p<<|].lz%=P;
t[p].lz=;
}
void pushup(int p)
{
t[p].sum=(t[p<<].sum+t[p<<|].sum)%P;
}
void build(int p,int l,int r)
{
t[p].l=l,t[p].r=r,t[p].lz=;
if (l!=r)
{
int mid=l+r>>;
build(p<<,l,mid);
build(p<<|,mid+,r);
pushup(p);
}
else
t[p].sum=val[indx[l]]%P;
}
void modify(int p,int l,int r,int w)
{
if (t[p].l==l && t[p].r==r)
{
t[p].sum+=w*(t[p].r-t[p].l+);
t[p].lz+=w;
return ;
}
pushdown(p);
int mid=t[p].l+t[p].r>>;
if (r<=mid) modify(p<<,l,r,w);
else if (l>mid) modify(p<<|,l,r,w);
else modify(p<<,l,mid,w),modify(p<<|,mid+,r,w);
pushup(p);
}
ll query(int p,int l,int r)
{
if (t[p].l==l && t[p].r==r)
return t[p].sum%P;
int mid=t[p].l+t[p].r>>;
pushdown(p);
if (r<=mid) return query(p<<,l,r)%P;
if (l>mid) return query(p<<|,l,r)%P;
return (query(p<<,l,mid)+query(p<<|,mid+,r))%P;
}
void pathInc(int u,int v,int w)
{
while (top[u]!=top[v])
{
if (deep[top[u]]<deep[top[v]]) swap(u,v);
modify(,pos[top[u]],pos[u],w);
u=fa[top[u]];
}
if (deep[u]>deep[v]) swap(u,v);
modify(,pos[u],pos[v],w);
}
ll pathQuery(int u,int v)
{
ll ret=;
while (top[u]!=top[v])
{
if (deep[top[u]]<deep[top[v]]) swap(u,v);
ret=(ret+query(,pos[top[u]],pos[u]))%P;
u=fa[top[u]];
}
if (deep[u]>deep[v]) swap(u,v);
return (ret+query(,pos[u],pos[v]))%P;
}
int main()
{
scanf("%d%d%d%lld",&n,&m,&r,&P);
for (int i=;i<=n;i++)
scanf("%lld",&val[i]);
for (int i=,u,v;i<n;i++)
scanf("%d%d",&u,&v),add(u,v);
dfs1(r,,);
dfs2(r,r);
build(,,n);
for (int i=,x,y,z;i<=m;i++)
{
scanf("%d",&op);
if (op==)
scanf("%d%d%d",&x,&y,&z),pathInc(x,y,z);
else if (op==)
scanf("%d%d",&x,&y),printf("%lld\n",pathQuery(x,y));
else if (op==)
scanf("%d%d",&x,&z),modify(,pos[x],back[x],z);
else scanf("%d",&x),printf("%lld\n",query(,pos[x],back[x]));
}
return ;
}

洛谷树剖模板题 P3384 | 树链剖分的更多相关文章

  1. 洛谷 P3373 【模板】线段树 2

    洛谷 P3373 [模板]线段树 2 洛谷传送门 题目描述 如题,已知一个数列,你需要进行下面三种操作: 将某区间每一个数乘上 xx 将某区间每一个数加上 xx 求出某区间每一个数的和 输入格式 第一 ...

  2. 洛谷P3372 【模板】线段树 1

    P3372 [模板]线段树 1 153通过 525提交 题目提供者HansBug 标签 难度普及+/提高 提交  讨论  题解 最新讨论 [模板]线段树1(AAAAAAAAA- [模板]线段树1 洛谷 ...

  3. 洛谷P3373 【模板】线段树 2

     P3373 [模板]线段树 2 47通过 186提交 题目提供者HansBug 标签 难度提高+/省选- 提交  讨论  题解 最新讨论 为啥WA(TAT) 题目描述 如题,已知一个数列,你需要进行 ...

  4. 洛谷 P3373 【模板】线段树 2 解题报告

    P3373 [模板]线段树 2 题目描述 如题,已知一个数列,你需要进行下面三种操作: 1.将某区间每一个数乘上\(x\) 2.将某区间每一个数加上\(x\) 3.求出某区间每一个数的和 输入输出格式 ...

  5. 洛谷 P3372 【模板】线段树 1

    P3372 [模板]线段树 1 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.求出某区间每一个数的和 输入输出格式 输入格式: 第一行包含两个整数N.M,分别 ...

  6. 洛谷—— P3372 【模板】线段树 1

    P3372 [模板]线段树 1 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.求出某区间每一个数的和 输入输出格式 输入格式: 第一行包含两个整数N.M,分别 ...

  7. 洛谷——P3373 【模板】线段树 2&& B 数据结构

    P3373 [模板]线段树 2 题目描述 如题,已知一个数列,你需要进行下面三种操作: 1.将某区间每一个数乘上x 2.将某区间每一个数加上x 3.求出某区间每一个数的和 线段树维护区间乘法 1.如何 ...

  8. 洛谷 P3373 【模板】线段树 2 题解

    P3373 [模板]线段树 2 题目描述 如题,已知一个数列,你需要进行下面三种操作: 1.将某区间每一个数乘上x 2.将某区间每一个数加上x 3.求出某区间每一个数的和 输入格式 第一行包含三个整数 ...

  9. 洛谷P3373 【模板】线段树 2 && P2023 [AHOI2009]维护序列——题解

    题目传送: P3373 [模板]线段树 2  P2023 [AHOI2009]维护序列 该题较传统线段树模板相比多了一个区间乘的操作.一提到线段树的区间维护问题,就自然想到了“懒标记”:为了降低时间复 ...

随机推荐

  1. BZOJ3884: 上帝与集合的正确用法(欧拉函数 扩展欧拉定理)

    Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 3860  Solved: 1751[Submit][Status][Discuss] Descripti ...

  2. gd库 给底图写入文字问题

    png-8的图片 设置颜色会返回false,换成png-24的就可以了

  3. ES6笔记03-Set和Map数据结构

    ES6提供了新的数据结构Set.它类似于数组,但是成员的值都是唯一的,没有重复的值.Set本身是一个构造函数,用来生成Set数据结构. var s = new Set(); [2, 3, 5, 4, ...

  4. Apache2服务配置ubuntu16.04+django1.11

    话不多说直接上步骤 环境 Ubuntu 16.04 Python 3.5.2 Django 1.11 Apache 2.4 1.Apache2安装 sudo apt-get install apach ...

  5. python分布式爬虫--房天下

    第一步安装redis redis在windows系统中的安装与启动: 下载:redis官方是不支持windows操作系统的.但是微软的开源部门将redis移植到了windows上.因此下载地址不是在r ...

  6. C# 设置窗口大小为不可调、取消最大化、最小化窗口按键

    this.FormBorderStyle = FormBorderStyle.FixedDialog;//设置边框为不可调节 this.MaximizeBox = false; //取消最大化按键 t ...

  7. 分离链接法(Separate Chaining)

    之前我们说过,对于需要动态维护的散列表 冲突是不可避免的,无论你的散列函数设计的有多么精妙.因此我们解决的重要问题就是:一旦发生冲突,我们该如何加以排解? 我们在这里讨论最常见的两种方法:分离链接法和 ...

  8. PHP.TP框架下商品项目的优化1-时间插件、鼠标所在行高亮、布局规划页面

    1.优化搜索表单中按时间搜索的功能 添加一个时间插件datetimepicker,在lst.html中,注意要导入jquery.min.js,此处从前文的在线编辑器中导入 <!-- 导入 --& ...

  9. 10,python开发之virtualenv与virtualenvwrapper

      在使用 Python 开发的过程中,工程一多,难免会碰到不同的工程依赖不同版本的库的问题: 亦或者是在开发过程中不想让物理环境里充斥各种各样的库,引发未来的依赖灾难. 此时,我们需要对于不同的工程 ...

  10. P2341 [HAOI2006]受欢迎的牛(tarjan+缩点)

    P2341 [HAOI2006]受欢迎的牛 题目描述 每头奶牛都梦想成为牛棚里的明星.被所有奶牛喜欢的奶牛就是一头明星奶牛.所有奶 牛都是自恋狂,每头奶牛总是喜欢自己的.奶牛之间的“喜欢”是可以传递的 ...