这道题明明没有省选难度啊,为什么就成紫题了QAQ

另:在CF上A了但是洛谷Remote Judge玄学爆零。


思路是DFS序+线段树。

首先这道题直观上可以对于每一次修改用DFS暴力O(n),然后对于询问O(1)解决。

但是这个方法实在是太耗时间了,因此我们想到了dfs序。

所谓dfs序,就是按照dfs(这里我们用先序遍历)的顺序给这颗树打上一个标签。

然后我们就可以把这颗树“拍平”,用一些支持区间修改单点查询的数据结构log级别解决问题了。


当然这样粗略地说一遍肯定会有人看不懂,还是通过一个实例讲解好一点。

举个例子,我们有这样一棵树:

每个节点都有一个编号。现在,我们按照dfs的顺序将这颗树写出来,也就是:

这样这颗树已经被我们“拍平”了,但是仍然无法解题。

为什么?

因为对于每一颗子树,你只知道它从什么地方开始,不知道它在什么地方结束。

解决方案很简单,我们多记录一个out,表示回溯的时候的顺序,这样就可以解决问题了。

dfs这个部分的代码如下:

 void dfs(int x){
in[x]=++dfn; //in是子树的起点。
for(int i=head[x];i;i=edge[i].next){
int y=edge[i].to;
if(father[x]==y)continue;//father数组储存节点的父亲。(废话)
dep[y]=dep[x]+,father[y]=x,dfs(y);//dep数组储存节点的深度,这个数组的必要性我们后面会提到。
}
out[x]=dfn; //out是子树的中点。
}

然后现在考虑怎么做这道题。

很显然,最大的难点在于每次更新对于每一层节点改变的值都不一样。

等等,每一层?

对的,可以发现,相邻层的节点变化值互为相反数,而相隔层的节点变化值相同。

如果想不出解决方案这道题巨难,但如果想出来了就是一道水题。

很简单,线段树维护节点的变化值,然后在更新时我们对于层数为奇数的节点加上变化值,对于层数为偶数的节点减去变化值。

这样层数为奇数的节点与层数为偶数的节点变化量肯定是反的,也就符合题意。

实现是这样的:

 scanf("%d%d",&op,&x);
if(op==)scanf("%d",&y),add(,in[x],out[x],dep[x]%?y:-y);
else printf("%d\n",a[x]+query(,x)*(dep[x]%?:-));

这个玩意的正确性很好说明,自己模拟一下就OK了。

------------

总的来说,这道题就是敲个模板。

代码如下:

 #include<iostream>
#include<cstdio>
#define ls p<<1
#define rs p<<1|1
using namespace std;
const int N=;
int n,m,v,u,cnt,op,x,dfn,y;
int a[N],in[N],head[N],dep[N],out[N],father[N];
struct node{int to,next;}edge[N];
inline void add(int a,int b){edge[++cnt].to=b,edge[cnt].next=head[a],head[a]=cnt;}
struct tnode{int add,sum,l,r;}tree[N<<];
void dfs(int x){
in[x]=++dfn;
for(int i=head[x];i;i=edge[i].next){
int y=edge[i].to;
if(father[x]==y)continue;
dep[y]=dep[x]+,father[y]=x,dfs(y);
}
out[x]=dfn;
}
inline void pushup(int p){
tree[p].sum=tree[ls].sum+tree[rs].sum;
}
inline void pushdown(int p,int l,int r){
if(!tree[p].add)return;
int mid=(l+r)>>;
tree[ls].add+=tree[p].add;tree[rs].add+=tree[p].add;
tree[ls].sum+=tree[p].add*(mid-l+);tree[rs].sum+=tree[p].add*(r-mid);
tree[p].add=;
}
void build(int p,int l,int r){
tree[p].l=l,tree[p].r=r;
tree[p].add=tree[p].sum=;
if(l==r)return;
int mid=(l+r)>>;
build(ls,l,mid);build(rs,mid+,r);
pushup(p);
}
void add(int p,int l,int r,int val){
if(l<=tree[p].l&&tree[p].r<=r){tree[p].add+=val;tree[p].sum+=(tree[p].r-tree[p].l+)*val;return;}
int mid=(tree[p].l+tree[p].r)>>;
pushdown(p,tree[p].l,tree[p].r);
if(l<=mid)add(ls,l,r,val);
if(r>mid)add(rs,l,r,val);
pushup(p);
}
int query(int p,int x){
if(tree[p].l==tree[p].r)return tree[p].sum;
int mid=(tree[p].l+tree[p].r)>>;
pushdown(p,tree[p].l,tree[p].r);
if(x<=mid)return query(ls,x);
else return query(rs,x);
}
int main(){
scanf("%d%d",&n,&m);
for(int i=;i<=n;++i)scanf("%d",&a[i]);
for(int i=;i<=n-;++i){
scanf("%d%d",&v,&u);
add(v,u);
}
dfs();
build(,,n);
while(m--){
scanf("%d%d",&op,&x);
if(op==)scanf("%d",&y),add(,in[x],out[x],dep[x]%?y:-y);
else printf("%d\n",a[x]+query(,x)*(dep[x]%?:-));
}
return ;
}

题解 CF383C 【Propagating tree】的更多相关文章

  1. 「CF383C Propagating tree」

    这应该属于一个比较麻烦的数据结构处理树上问题. 题目大意 给出一颗根节点编号为 \(1\) 的树,对于一个节点修改时在它的子树中对于深度奇偶性相同的节点加上这个权值,不同则减去这个值,单点查询. 分析 ...

  2. CF383C Propagating tree (线段树,欧拉序)

    \(tag\)没开够\(WA\)了一发... 求出\(dfs\)序,然后按深度分类更新与查询. #include <iostream> #include <cstdio> #i ...

  3. Codeforces Round #225 (Div. 1) C. Propagating tree dfs序+树状数组

    C. Propagating tree Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/383/p ...

  4. Codeforces Round #225 (Div. 1) C. Propagating tree dfs序+ 树状数组或线段树

    C. Propagating tree Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/383/p ...

  5. Codeforces Round #225 (Div. 2) E. Propagating tree dfs序+-线段树

    题目链接:点击传送 E. Propagating tree time limit per test 2 seconds memory limit per test 256 megabytes inpu ...

  6. 【题解】Digit Tree

    [题解]Digit Tree CodeForces - 716E 呵呵以为是数据结构题然后是淀粉质还行... 题目就是给你一颗有边权的树,问你有多少路径,把路径上的数字顺次写出来,是\(m\)的倍数. ...

  7. 【题解】[P4178 Tree]

    [题解]P4178 Tree 一道点分治模板好题 不知道是不是我见到的题目太少了,为什么这种题目都是暴力开值域的桶QAQ?? 问点对,考虑点分治吧.直接用值域树状数组开下来,统计的时候直接往树状数组里 ...

  8. AC日记——Propagating tree Codeforces 383c

    C. Propagating tree time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  9. CodeForces 383C Propagating tree

    Propagating tree Time Limit: 2000ms Memory Limit: 262144KB This problem will be judged on CodeForces ...

  10. C. Propagating tree

    C. Propagating tree time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

随机推荐

  1. oracle查询优化之子查询条件优化

    环境:oracle 11g 现有a表与b表通过a01字段关联,要查询出a表的数据在b表没有数据的数据:sql如下 ) ) 因为flag是虚拟字段没有走不了索引导致这条sql执行起来特别慢 310W条数 ...

  2. 原生的ajax请求----(播放托管到爱奇艺上的视频)

    播放视频 $(function(){ //视频播放 $('.play-icon').click(function () { $.ajax({ type:"get", url: &q ...

  3. 洛谷 P1403 [AHOI2005]约数研究

    怎么会有这么水的省选题 一定是个签到题. 好歹它也是个省选题,独立做出要纪念一下 很容易发现在1~n中,i的因子数是n / i 那就枚举每一个i然后加起来就OK了 #include<cstdio ...

  4. idea 包的显示方式

    idea 可以通过点击Project的导航栏里的小齿轮里面有一个 Flatten packages 选项,将其勾上.就可以得到跟eclipse一样的包的显示方式. 没有设置默认是这样的 2018-06 ...

  5. [Test] Easy automated testing in NodeJS with TestCafe

    Quickly get up and running with sensible automated testing scenarios written in ES6. Installing and ...

  6. 严重: 文档无效: 找不到语法。 at (null:2:19)

    1.错误描写叙述 严重: 文档无效: 找不到语法. at (null:2:19) org.xml.sax.SAXParseException; systemId: file:/D:/MyEclipse ...

  7. IOS音频架构之Audio Unit

    在前面的章节部分我们已经对IOS音频结构有了一个清晰的认识,知道Audio Unit是位于整个音频结构的最底层,这一层非常多API已经開始和硬件打交道了.所以比較复杂,有了前面的基础再来看这个部分就比 ...

  8. layer是什么

    layer是什么 总结 layer就是一个web弹框 简介 layer是一款web弹层组件,致力于服务各个水平段的开发人员. 可以让你想到即可做到的web弹窗 概述 [1]  layer,一个可以让你 ...

  9. 木马——本质就是cs socket远程控制,反弹木马是作为c端向外发起网络请求

    摘自:http://kczxsp.hnu.edu.cn/upload/20150504165623705.pdf 里面对于木马的实验过程写得非常清楚,值得一看.   木马是隐藏在正常程序中的具有特殊功 ...

  10. zzulioj--1828-- 贪心的小猫咪(贪心模拟)

    1828: 贪心的小猫咪 Time Limit: 1 Sec  Memory Limit: 128 MB Submit: 14  Solved: 4 SubmitStatusWeb Board Des ...