原题链接

对于以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 ;
}

最新文章

  1. ASP.NET Aries 开源开发框架:开发指南(一)
  2. [转] Fix: Screen Clipping Shortcut In OneNote Not Working After Upgrading To Windows 8.1
  3. 汉诺塔(c++)
  4. 在idea中mybatis错误(1)
  5. HDU 4471 矩阵快速幂 Homework
  6. &lt;!--[if IE]&gt;….&lt;![endif]--&gt; (&lt;!--[if !IE]&gt;||&lt;![endif]--&gt;)的用法
  7. 关于Memo或者Edit之类控件, 直接设置Text无法撤销的解决方案
  8. UITextField的常用属性,Delegate,重绘
  9. dede操作成功信息提示修改
  10. Flash,EEPROM差别
  11. 从零开始系列之vue全家桶(1)安装前期准备nodejs+cnpm+webpack+vue-cli+vue-router
  12. 【原创】大叔经验分享(44)hdfs副本数量
  13. 一些matlab命令
  14. pygame-KidsCanCode系列jumpy-part11-角色动画(下)
  15. java线程学习之线程创建
  16. centos7防火墙管理的变化
  17. Leetcode——53.最大子序和
  18. 修改ECSHOP的小数点保留位数
  19. hadoop 启动的时候datanode报错 Problem connecting to server
  20. linux虚拟化课堂总结图2019_4_23

热门文章

  1. Scott Young-《如何高效学习》
  2. Cannot resolve reference to bean &#39;sessionFactory&#39; while setting bean property &#39;sessionFactory&#39;; 没有sessionFactory
  3. MySQL选择的执行计划性能底下原因分析--实战案例分析
  4. python基础之模块part1
  5. Android面试收集录10 LruCache原理解析
  6. salt demo 环境
  7. 05,Python网络爬虫之三种数据解析方式
  8. 利用 ESLint 检查代码质量
  9. Windows usb设备正在使用中
  10. 用scrapy数据抓取实践