题解 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 ...
随机推荐
- hiho1080 - 数据结构 线段树(入门题,两个lazy tag)
题目链接 维护区间和,两个操作:一个是将某个区间设置成一个值,一个是将某个区间增加一个固定值 /**************************************************** ...
- 3ds Max 2018 在安装后无法启动或出现不稳定
问题: 安装 3ds Max 2018 后,软件无法正常启动,或在打开后不久出现不稳定和崩溃. 原因: 有多种原因可能会导致这些错误: ▪ 3ds Max.Windows 更新和 ProSound.d ...
- HDU-1043 Eight八数码 搜索问题(bfs+hash 打表 IDA* 等)
题目链接 https://vjudge.net/problem/HDU-1043 经典的八数码问题,学过算法的老哥都会拿它练搜索 题意: 给出每行一组的数据,每组数据代表3*3的八数码表,要求程序复原 ...
- BZOJ 2141 排队(CDQ分治)
我们把每一次交换看做两个插入两个删除.然后就是一个三维偏序.时间一维,下标一维,权值一维. #include<iostream> #include<cstring> #incl ...
- [NOI2002]贪吃的九头龙(树形dp)
[NOI2002]贪吃的九头龙 题目背景 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是 说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的 ...
- 洛谷 P3147 [USACO16OPEN]262144
P3147 [USACO16OPEN]262144 题目描述 Bessie likes downloading games to play on her cell phone, even though ...
- Qt之水平/垂直布局(QBoxLayout、QHBoxLayout、QVBoxLayout)
简述 QBoxLayout可以在水平方向或垂直方向上排列控件,由QHBoxLayout.QVBoxLayout所继承. QHBoxLayout:水平布局,在水平方向上排列控件,即:左右排列. QVBo ...
- Dubbo分布式服务框架入门(附project)
要想了解Dubbo是什么,我们不防先了解它有什么用. 使用场景:比方我想开发一个网上商城项目.这个网上商城呢,比較复杂.分为pc端web管理后台.微信端销售公众号,那么我们分成四个项目,pc端站点,微 ...
- RvmTranslator7.0-OBJ
RvmTranslator7.0-OBJ eryar@163.com RvmTranslator can translate the RVM file exported by AVEVA Plant( ...
- CG 内置函数 和 HLSL 内置函数
CG 内置函数 英伟达官网链接: http://http.developer.nvidia.com/Cg/index_stdlib.html absacosallanyasinatan2atanbi ...