题解 CF383C 【Propagating tree】
这道题明明没有省选难度啊,为什么就成紫题了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】的更多相关文章
- 「CF383C Propagating tree」
这应该属于一个比较麻烦的数据结构处理树上问题. 题目大意 给出一颗根节点编号为 \(1\) 的树,对于一个节点修改时在它的子树中对于深度奇偶性相同的节点加上这个权值,不同则减去这个值,单点查询. 分析 ...
- CF383C Propagating tree (线段树,欧拉序)
\(tag\)没开够\(WA\)了一发... 求出\(dfs\)序,然后按深度分类更新与查询. #include <iostream> #include <cstdio> #i ...
- 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 ...
- 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 ...
- 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 ...
- 【题解】Digit Tree
[题解]Digit Tree CodeForces - 716E 呵呵以为是数据结构题然后是淀粉质还行... 题目就是给你一颗有边权的树,问你有多少路径,把路径上的数字顺次写出来,是\(m\)的倍数. ...
- 【题解】[P4178 Tree]
[题解]P4178 Tree 一道点分治模板好题 不知道是不是我见到的题目太少了,为什么这种题目都是暴力开值域的桶QAQ?? 问点对,考虑点分治吧.直接用值域树状数组开下来,统计的时候直接往树状数组里 ...
- AC日记——Propagating tree Codeforces 383c
C. Propagating tree time limit per test 2 seconds memory limit per test 256 megabytes input standard ...
- CodeForces 383C Propagating tree
Propagating tree Time Limit: 2000ms Memory Limit: 262144KB This problem will be judged on CodeForces ...
- C. Propagating tree
C. Propagating tree time limit per test 2 seconds memory limit per test 256 megabytes input standard ...
随机推荐
- 虚拟机CentOS6.8下安装JDK
CentOS6.8下 首先下载JDK,执行命令如下: wget http://download.oracle.com/otn-pub/java/jdk/8u172-b11/a58eab1ec24242 ...
- “.”开头,以"}"结尾,中间是任意字符的正则
"."开头,以"}"结尾,中间是任意字符的正则 /^\..+\{$/
- (WC2016模拟十八)【BZOJ4299】[CodeChef]FRBSUM
咕了若干天我终于来补坑了qwq HINT $1\leq N,M\leq 10^5$ $1\leq \sum A_i\leq 10^9$ 题解: 虽然场上做出来了但还是觉得好神啊! 假设当前集合能凑出$ ...
- [Debug]SpaceVim中neomake报错 Error while trying to load a compilation database
回家装上archlinux,突发奇想装个SpaceVim写题 安装配置一路可以说是没有太大问题 最后在写题时出现如下问题 Error while trying to load a compilatio ...
- Springboot - -web应用开发-Servlets, Filters, listeners
一.Web开发使用 Controller 基本上可以完成大部分需求,但是我们还可能会用到 Servlet. Filter. Listener等等 二.在spring boot中的三种实现方式 方法一: ...
- scrapy爬取boss直聘实习生数据
这个..是我最近想找实习单位..结果发现boss上很多实习单位名字就叫‘实习生’.......太不讲究了 == 难怪一直搜不到..咳,其实是我自己水平有限,有些简历根本就投不出去 == 所以就想爬下b ...
- 怎么给Unity写一个原生的插件
本文章由cartzhang编写,转载请注明出处. 所有权利保留. 文章链接:http://blog.csdn.net/cartzhang/article/details/50266889 作者:car ...
- Python seed() 函数--每次产生一样的随机数系列
import random random.seed( 10 ) print("Random number with seed 10 : ", random.random()) #0 ...
- jsp urlrewrite 中正則表達式不包括某个字符串写法
因在程序中须要做城市间跳转,可是页面中包括的css.scripts和图片等路径是要排除在外的. 这就须要在正则中指定当遇到哪些 字符时须要略过. 正则例如以下: /((? !css)(?!script ...
- MapReduce中combine、partition、shuffle的作用是什么
http://www.aboutyun.com/thread-8927-1-1.html Mapreduce在hadoop中是一个比較难以的概念.以下须要用心看,然后自己就能总结出来了. 概括: co ...