题目:

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

操作 1 :把某个节点 x 的点权增加 a 。

操作 2 :把某个节点 x 为根的子树中所有点的点权都增加 a 。

操作 3 :询问某个节点 x 到根的路径中所有点的点权和。

题解:

ysf : 树链剖分大水题 !

lrd : 直接树剖不就好了嘛

gls : 这树剖不行吗 ?

lyc : ... ...

本人表示,当时真的脑袋抽了,忘了树剖就能搞...

但是我脑袋这么一抽就想出了一个\(O(nlogn)\)的解法.

我们处理出dfs序.

如图:



对应dfs序为:

\(1,2,3,-3,4,-4,-2,5,6,-6,-5,-1\)

然后对于所有的单点修改修改相应的入栈点和出栈点

对于一个子树修改就修改左端点为入栈序,右端点为出栈序的区间

对于一个查询就直接查询区间[1,入栈序]的权值和.

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
inline void read(ll &x){
x=0;char ch;bool flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
const ll maxn = 210010;
struct Edge{
ll to,next;
}G[maxn<<1];
ll head[maxn],cnt;
inline void add(ll u,ll v){
G[++cnt].to = v;
G[cnt].next = head[u];
head[u] = cnt;
}
#define v G[i].to
ll ind[maxn],dfs_clock,oud[maxn],idx[maxn];
void dfs(ll u,ll f){
ind[u] = ++ dfs_clock;
idx[dfs_clock] = u;
for(ll i = head[u];i;i=G[i].next){
if(v == f) continue;
dfs(v,u);
}
oud[u] = ++ dfs_clock;
idx[dfs_clock] = u;
}
#undef v
struct Node{
Node *ch[2];
ll val,lazy;
ll num1,num2;
void update(){
val = ch[0]->val + ch[1]->val;
num1 = ch[0]->num1 + ch[1]->num1;
num2 = ch[0]->num2 + ch[1]->num2;
}
}*null,mem[maxn<<2],*it,*root;
inline void init(){
it = mem;null = it++;
null->ch[0] = null->ch[1] = null;
null->val = null->lazy = null->num1 = null->num2 = 0;
root = null;
}
inline Node* newNode(){
Node *p = it++;p->ch[0] = p->ch[1] = null;
return p;
}
inline void pushdown(Node *p){
if(p == null || p->lazy == 0) return ;
if(p->ch[0] != null){
p->ch[0]->val += p->ch[0]->num1*p->lazy - p->ch[0]->num2*p->lazy;
p->ch[0]->lazy += p->lazy;
}
if(p->ch[1] != null){
p->ch[1]->val += p->ch[1]->num1*p->lazy - p->ch[1]->num2*p->lazy;
p->ch[1]->lazy += p->lazy;
}
p->lazy = 0;
}
inline void insert(Node* &p,ll l,ll r,ll pos,ll val,ll ty){
if(p == null) p = newNode();
if(l == r){
if(ty == 1) p->num1 = 1,p->num2 = 0;
else if(ty == -1) p->num1 = 0,p->num2 = 1;
p->val = p->num1*val - p->num2*val;
return ;
}
ll mid = l+r >> 1;
if(pos <= mid) insert(p->ch[0],l,mid,pos,val,ty);
else insert(p->ch[1],mid+1,r,pos,val,ty);
p->update();
}
inline void modify(Node *p,ll l,ll r,ll L,ll R,ll val){
if(L <= l &&r <= R){
p->lazy += val;
p->val += p->num1*val - p->num2*val;
return ;
}
pushdown(p);
ll mid = l+r >> 1;
if(L <= mid) modify(p->ch[0],l,mid,L,R,val);
if(R > mid) modify(p->ch[1],mid+1,r,L,R,val);
p->update();
}
inline ll query(Node *p,ll l,ll r,ll L,ll R){
if(L <= l && r <= R) return p->val;
pushdown(p);ll mid = l+r >> 1;
if(R <= mid) return query(p->ch[0],l,mid,L,R);
if(L > mid) return query(p->ch[1],mid+1,r,L,R);
return query(p->ch[0],l,mid,L,R) + query(p->ch[1],mid+1,r,L,R);
}
ll c[maxn];
int main(){init();
ll n,m;read(n);read(m);
for(ll i=1;i<=n;++i) read(c[i]);
for(ll i=1,u,v;i<n;++i){
read(u);read(v);
add(u,v);add(v,u);
}dfs(1,1);
for(ll i=1;i<=n;++i){
insert(root,1,dfs_clock,ind[i],c[i],1);
insert(root,1,dfs_clock,oud[i],c[i],-1);
}
watch(root,1,dfs_clock);
ll op,x,a;
while(m--){
read(op);
if(op == 1){
read(x);read(a);
modify(root,1,dfs_clock,ind[x],ind[x],a);
modify(root,1,dfs_clock,oud[x],oud[x],a);
}else if(op == 2){
read(x);read(a);
modify(root,1,dfs_clock,ind[x],oud[x],a);
}else if(op == 3){
read(x);
ll ans = query(root,1,dfs_clock,1,ind[x]);
printf("%lld\n",ans);
}
}
getchar();getchar();
return 0;
}

bzoj 4034: 树上操作 线段树的更多相关文章

  1. [BZOJ 4034] 树上操作

    Link: BZOJ 4034 传送门 Solution: 树剖模板题…… Code: #include <bits/stdc++.h> using namespace std; type ...

  2. BZOJ 4034 树上操作(树的欧拉序列+线段树)

    刷个清新的数据结构题爽一爽? 题意: 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x ...

  3. BZOJ 4034 [HAOI2015]树上操作 线段树+树剖或dfs

    题意 直接照搬原题面 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子树中所 ...

  4. BZOJ 4034"树上操作"(DFS序+线段树)

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

  5. bzoj 4034(DFS序+线段树)

    这个题多了一个操作难度直线上升,看完题解才会写 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个操作,分为三种:操作 1 :把某个节点 x 的点权增加 a .操作 2 :把某个节点 ...

  6. cf1110F 离线+树上操作+线段树区间更新

    自己搞的算法超时了..但是思路没什么问题:用线段树维护每个点到叶子节点的距离即可 /* 线段树维护区间最小值,每次向下访问,就把访问到的点对应的区间段减去边权 到另一颗子树访问时,向上回溯时加上减去的 ...

  7. bzoj 2962 序列操作——线段树(卷积?)

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2962 如果 _,_,_,…… 变成了 (_+k),(_+k),(_+k),…… ,计算就是在 ...

  8. bzoj 2962 序列操作 —— 线段树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2962 维护 sum[i] 表示选 i 个的乘积和,合并两个子树就枚举两边选多少,乘起来即可: ...

  9. [BZOJ3307] 雨天的尾巴(树上差分+线段树合并)

    [BZOJ3307] 雨天的尾巴(树上差分+线段树合并) 题面 给出一棵N个点的树,M次操作在链上加上某一种类别的物品,完成所有操作后,要求询问每个点上最多物品的类型. N, M≤100000 分析 ...

随机推荐

  1. iOS-Core-Animation-Advanced-Techniques(二)

    本文转载至 http://www.cocoachina.com/ios/20150104/10816.html 视觉效果和变换 (四)视觉效果 嗯,园和椭圆还不错,但如果是带圆角的矩形呢? 我们现在能 ...

  2. java中业务接口

    今天写完业务层在抽取接口的时候脑子里突然产生了一个问题:抽取接口到底有什么用呢? 在刚刚学习接口的时候知道接口是为了实现java的多继承,但是现在每一个业务类都要抽取一个接口,每当该类需要增加方法的时 ...

  3. Entity Framework 4.1:复杂类型

    这篇文章将讨论复杂类型. 默认情况下,EF4.1 将类映射到表,这是约定,但是有时候,我们需要模型比表的粒度更细一些. 地址是一个典型的例子,看一下下面的客户类. )] publicstring St ...

  4. js绑定键盘enter事件

    js写法: document.onkeydown = function(event){ var event=document.all?window.event:event; if((event.key ...

  5. 9.Django里的数据同步migrations命令

    一个关键的目录: 目录名:migrations 作用:用来存放通过makemigrations命令生成的数据库脚本,这里的内容一般不要手动去改 规定:app目录下必须要有migrations目录且目录 ...

  6. vim 光标的移动和跳转文件的位置

    刚启动vim进入的就是命令模式 在命令模式下 h等于左箭头 j等于下箭头 k等于上箭头 l等于右箭头 想要多次移动可以使用30j或30↓向下移动30行 在命令模式下输入0跳到行头 在命令模式下输入$跳 ...

  7. Mysql——JDBC编程 理论介绍

    一.JDBC简介(来自俞琰--数据库老师) Java数据库编程主要使用JDBC技术.JDBC是一种用于执行SQL语句的Java API.它由一组用Java编写的类和接口组成.JDBC为开发人员提供了一 ...

  8. 0521 HTML基础

    一.web标准 web准备介绍: w3c:万维网联盟组织,用来制定web标准的机构(组织) web标准:制作网页遵循的规范 web准备规范的分类:结构标准.表现标准.行为标准. 结构:html.表示: ...

  9. P4455 [CQOI2018]社交网络(矩阵树定理)

    题目 P4455 [CQOI2018]社交网络 \(CQOI\)的题都这么裸的吗?? 做法 有向图,指向叶子方向 \(D^{out}(G)-A(G)\) 至于证明嘛,反正也就四个定理,先挖个坑,省选后 ...

  10. 日期类(C++实现)

    //-------------------------------------------------------------------------- /* **功能:实现日期的简单操作 ** ** ...