P3178 [HAOI2015]树上操作

题目链接:https://www.luogu.org/problemnew/show/P3178

题目描述

有一棵点数为 N 的树,以点 1 为根,且树点有边权。然后有 M 个操作,分为三种:

  • 操作 1 :把某个节点 x 的点权增加 a 。
  • 操作 2 :把某个节点 x 为根的子树中所有点的点权都增加 a 。
  • 操作 3 :询问某个节点 x 到根的路径中所有点的点权和。

输入输出格式

输入格式:

第一行包含两个整数 N, M 。表示点数和操作数。
接下来一行 N 个整数,表示树中节点的初始权值。
接下来 N-1 行每行两个正整数 from, to , 表示该树中存在一条边 (from, to) 。
再接下来 M 行,每行分别表示一次操作。其中第一个数表示该操作的种类( 1-3 ) ,之后接这个操作的参数( x 或者 x a ) 。

输出格式:

对于每个询问操作,输出该询问的答案。答案之间用换行隔开。

输入输出样例

输入样例#1:

5 5
1 2 3 4 5
1 2
1 4
2 3
2 5
3 3
1 2 1
3 5
2 1 2
3 3
输出样例#1:

6
9
13

题解:

这似乎是个熟练剖分模板题= =然而蒟蒻还没学过熟练剖分,所以用dfs序+线段树写了下,就当熟悉一下这些比较基本的数据结构吧。

这题用dfs序+线段树还是比较巧妙的,首先这是子树的问题嘛,我们可以考虑求一下dfs序,并且每个结点对应了一个管辖的区间in[x]~out[x]。

然后分析题目中的操作,我们主要的分析是从操作对一条链的影响来分析的:

第一个操作单点更新,那么我们就可以知道,以当前点x为根的子树的所有点,其到根节点的距离都为增加,那么我们利用dfs序的性质,将in[x]更新一下就行了,这样前缀和也是会增加相应值的。

第二个操作子树更新,由于这个题我们考虑的是更新对链的影响,那么可以知道,这个子树上的结点受到的影响主要取决前面有多少个结点。这里我们还是要巧妙运用dfs序的性质,在对相应区间进行更新时,更新的值为区间中"+"的个数减去区间中"-"的个数。这样在查询前缀和时就能正确地统计出答案(yy一下就好了)。

第三个查询当前点到根节点这条链的权值和,这里我们可以直接根据dfs序的性质查询前缀和就行了。

代码如下(lazy标记没处理好查了半年错):

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll N = ;
ll n,m;
ll a[N];
struct Tree{
ll l,r;
ll lazy,sum;
}tre[N<<];
ll in[N],out[N],head[N],f[N<<];
ll c[N<<],num[N<<];
ll dfn,tot;
struct Edge{
ll u,v,next;
}e[N<<];
void adde(ll u,ll v){
e[tot].v=v;e[tot].next=head[u];head[u]=tot++;
}
void dfs(ll u,ll fa){
in[u]=++dfn;
num[dfn]=u;f[dfn]=;
for(ll i=head[u];i!=-;i=e[i].next){
ll v=e[i].v;
if(v!=fa) dfs(v,u);
}
out[u]=++dfn;
num[dfn]=u;f[dfn]=-;
}
void build(ll rt,ll l,ll r){
tre[rt].l=l;tre[rt].r=r;
ll mid=(l+r)>>;
if(l==r){
tre[rt].sum=f[l]*a[num[l]];
return ;
}
build(rt<<,l,mid);
build(rt<<|,mid+,r);
tre[rt].sum=tre[rt<<].sum+tre[rt<<|].sum;
}
void push_down(ll rt){
ll lazy=tre[rt].lazy;
tre[rt<<].sum+=lazy*(c[tre[rt<<].r]-c[tre[rt<<].l-]);
tre[rt<<|].sum+=lazy*(c[tre[rt<<|].r]-c[tre[rt<<|].l-]);
tre[rt<<].lazy+=lazy;
tre[rt<<|].lazy+=lazy;
tre[rt].lazy=;
return ;
}
void add(ll rt,ll id,ll z){
ll l=tre[rt].l,r=tre[rt].r;
if(l==id&&r==id){
tre[rt].sum+=z;
return ;
}
push_down(rt);
ll mid=(l+r)>>;
if(mid>=id) add(rt<<,id,z);
else add(rt<<|,id,z);
tre[rt].sum=tre[rt<<].sum+tre[rt<<|].sum;
return ;
}
void update(ll rt,ll l,ll r,ll z){
ll L=tre[rt].l,R=tre[rt].r;
if(l<=L && R<=r){
tre[rt].sum+=(c[R]-c[L-])*(ll)z;
tre[rt].lazy+=z;
return ;
}
if(tre[rt].lazy) push_down(rt);
ll mid=(L+R)>>;
if(l<=mid) update(rt<<,l,r,z);
if(r>mid) update(rt<<|,l,r,z);
tre[rt].sum=tre[rt<<].sum+tre[rt<<|].sum;
return ;
}
ll query(ll rt,ll l,ll r){
ll res = ;
ll L=tre[rt].l,R=tre[rt].r;
if(l<=L&&R<=r){
return tre[rt].sum;
}
if(tre[rt].lazy) push_down(rt);
ll mid=(L+R)>>;
if(l<=mid) res+=query(rt<<,l,r);
if(r>mid) res+=query(rt<<|,l,r);
return res ;
}
int main(){
scanf("%lld%lld",&n,&m);
for(ll i=;i<=n;i++) scanf("%lld",&a[i]);
memset(head,-,sizeof(head));
for(ll i=;i<n;i++){
ll u,v;
scanf("%lld%lld",&u,&v);
adde(u,v);adde(v,u);
}
dfs(,-);
for(ll i=;i<=dfn;i++) c[i]=c[i-]+f[i];
build(,,*n);
for(ll i=;i<=m;i++){
ll op,x,z;
scanf("%lld",&op);
if(op==){
scanf("%lld%lld",&x,&z);
add(,in[x],z);
add(,out[x],-z);
}else if(op==){
scanf("%lld%lld",&x,&z);
update(,in[x],out[x],z);
}else{
scanf("%lld",&x);
printf("%lld\n",query(,,in[x]));
}
}
return ;
}

洛谷P3178 [HAOI2015]树上操作(dfs序+线段树)的更多相关文章

  1. BZOJ4034 [HAOI2015]树上操作+DFS序+线段树

    参考:https://www.cnblogs.com/liyinggang/p/5965981.html 题意:是一个数据结构题,树上的,用dfs序,变成线性的: 思路:对于每一个节点x,记录其DFS ...

  2. 洛谷P3178 [HAOI2015]树上操作(线段树)

    题目描述 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个操作,分为三种:操作 1 :把某个节点 x 的点权增加 a .操作 2 :把某个节点 x 为根的子树中所有点的点权都增加 ...

  3. 洛谷P3178 [HAOI2015]树上操作 题解 树链剖分+线段树

    题目链接:https://www.luogu.org/problem/P3178 这道题目是一道树链剖分的模板题. 但是在解决这道问题的同事刷新了我的两个认识: 第一个认识是:树链剖分不光可以处理链, ...

  4. 洛谷——P3178 [HAOI2015]树上操作

    https://www.luogu.org/problem/show?pid=3178#sub 题目描述 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个操作,分为三种:操作 1 ...

  5. 洛谷P3178 [HAOI2015]树上操作

    题目描述 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个操作,分为三种:操作 1 :把某个节点 x 的点权增加 a .操作 2 :把某个节点 x 为根的子树中所有点的点权都增加 ...

  6. 洛谷 P3178 [HAOI2015]树上操作

    题目描述 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个操作,分为三种:操作 1 :把某个节点 x 的点权增加 a .操作 2 :把某个节点 x 为根的子树中所有点的点权都增加 ...

  7. [bzoj3306]树——树上倍增+dfs序+线段树

    Brief Description 您需要写一种数据结构,支持: 更改一个点的点权 求一个子树的最小点权 换根 Algorithm Design 我们先忽略第三个要求. 看到要求子树的最小点权,我们想 ...

  8. ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang(树上分块+dfs序+线段树)

    题意 链接:https://nanti.jisuanke.com/t/A1998 给出一个有根树(根是1),有n个结点.初始的时候每个结点的值都是0.下面有q个操作,操作有两种,操作1.将深度为L(根 ...

  9. 洛谷 3178 [HAOI2015]树上操作

    [题解] 就是个树链剖分的模板题. #include<cstdio> #include<algorithm> #include<cstring> #define L ...

随机推荐

  1. 【WXS】简要介绍说明

    WXS(WeiXin Script)是小程序的一套脚本语言. WXS有二种写法: 1) 以<wxs>标签书写脚本: 语法: <wxs module="[String]&qu ...

  2. python内建模块Collections

    # -*- coding:utf-8 -*- # OrderedDict可以实现一个FIFO(先进先出)的dict, # 当容量超出限制时,先删除最早添加的Key: from collections ...

  3. 【转】Haml 这货是啥? 附参考

    Haml是一种用来描述任何XHTML web document的标记语言,它是干净,简单的.而且也不用内嵌代码.Haml的职能就是替代那些内嵌代码的page page templating syste ...

  4. ionic typescript--验证码发送倒计时功能

    1.新建页面 ionic g page forget   2.mode.html文件 <ion-item> <ion-input clearInput [(ngModel)]='co ...

  5. 剑指offer-二叉树搜索树与双向链表25

    题目描述 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表.要求不能创建任何新的结点,只能调整树中结点指针的指向. class Solution: def Convert(self, pRo ...

  6. 剑指offer-顺时针打印矩阵19

    题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数 ...

  7. Wordcount -- MapReduce example -- Mapper

    Mapper maps input key/value pairs into intermediate key/value pairs. E.g. Input: (docID, doc) Output ...

  8. PHP正则相关

    描述字符串排列模式的一种自定义语法规则 如果可以使用字符串函数处理的任务 就不要使用正则 正则表达式 就是通过构建具有特定规则的模式,与输入的字符信息比较 在进行 分割 匹配 查找 替换 等工作   ...

  9. @ConfigurationProperties注解对数据的自动封装

    @ConfigurationProperties注解对数据的自动封装 @ConfigurationProperties可以对基本数据类型实现自动封装,可以封装格式为yyyy/MM/dd的日期 测试代码 ...

  10. 【算法分析】如何理解快慢指针?判断linked list中是否有环、找到环的起始节点位置。以Leetcode 141. Linked List Cycle, 142. Linked List Cycle II 为例Python实现

    引入 快慢指针经常用于链表(linked list)中环(Cycle)相关的问题.LeetCode中对应题目分别是: 141. Linked List Cycle 判断linked list中是否有环 ...