这道题明明没有省选难度啊,为什么就成紫题了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. 如何在Windows下安装Linux子系统(Ubuntu,openSUSU,SUSU Linux Server)

    注意:只有win10才能安装,安装的linux没有图形界面. 1.首先在win10设置 --> 更新与安装 --> 针对开发人员 ,选择开发人员模式. 2.win10 Cortana -- ...

  2. 新疆大学(新大)OJ xju 1010: 四个年级 C++ STL map 将4层循环优化成2层循环可解

    1010: 四个年级 时间限制: 3 Sec  内存限制: 128 MB 题目描述 为了加强全体软件学院的同学的较量,学院想让ACM协会设计一次活动让所以软件学院的同学都加入,并且要实现高年级和低年级 ...

  3. HDU 1548 A strange lift【BFS】

    题意:给出一个电梯,给出它的层数f,给出起点s,终点g,以及在每一层能够上或者下w[i]层,问至少需要按多少次按钮到达终点. 和POJ catch that cow一样,直接用了那一题的代码,发现一直 ...

  4. js判断浏览器的环境(pc端,移动端,还是微信浏览器)

    window.navigator.userAgent用来区分设备和浏览器 <!DOCTYPE html> <html> <head> <meta charse ...

  5. WordPress 增加 keywords 和 description

    WordPress 增加 keywords 和 description . <?php $keywords = '798资源网'; $description = '798资源网'; //文章页 ...

  6. Java基础学习总结(5)——多态

    一.面向对象最核心的机制--动态绑定,也叫多态 1.1.通过下面的例子理解动态绑定,即多态 package javastudy.summary; class Animal { /** * 声明一个私有 ...

  7. [转载]不唐突的JavaScript的七条准则

    经过多年的开发.教学和编写不唐突的JavaScript, 我发现了下面的一些准则.我希望它们可以帮助你对“为什么这样设计和执行JavaScript比较好”有一点理解.这些规则曾经帮助我更快地交付产品, ...

  8. Login.hbm.xml

    <?xml version="1.0" encoding="utf-8"?><!DOCTYPE hibernate-mapping PUBLI ...

  9. NIO框架之MINA源代码解析(一):背景

     "你们的agent占了好多系统的port.把我们的非常多业务系统都给整死了,给我们造成了非常大的损失.要求你们的相关领导下周过来道歉"   --   来自我们的一个客户.   ...

  10. 浅析为什么 char 类型的范围是 : -128~+127

    在 C 语言中. signed char 类型的范围为 -128~127,每本教科书上也这么写.可是没有哪一本书上(包含老师)也不会给你为什么是 -128~127,这个问题貌似看起来也非常easyea ...