题解 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 ...
随机推荐
- (WC2018模拟十二)【FJOI2016集训Day7T1】挑选子序列
题解: 这题太神仙了,先咕着,放个令人****的官方题解(精确覆盖问题是NP完全问题啊摔) Dancing Link X学习资料:QAQ TAT qwq Orz Donald Knuth大神... p ...
- Bedrock Linux 0.7.3 发布
Bedrock Linux是一种元分发,允许用户利用其他通常互斥的Linux发行版的功能,并让它们无缝地一起工作.该项目发布了其0.7.x系列,Bedrock Linux 0.7.3的更新. 新的更新 ...
- 关于 Error: No PostCSS Config found in 的错误
问题描述: 项目在本地运行不报错,上传到 GitHub 之后,再 clone 到本地, npm install 安装完成之后再执行 npm run dev 这时报错 Error: No PostCSS ...
- 洛谷 P1403 [AHOI2005]约数研究
怎么会有这么水的省选题 一定是个签到题. 好歹它也是个省选题,独立做出要纪念一下 很容易发现在1~n中,i的因子数是n / i 那就枚举每一个i然后加起来就OK了 #include<cstdio ...
- NYIST 1108 最低的惩罚
最低的惩罚 时间限制:3000 ms | 内存限制:65535 KB 难度:3 描述 那么现在问题就来了... 给你N(1=<N<=15)个任务,每个任务有一个截止完成时间t(1= ...
- 基于SpringMVC+Bootstrap+DataTables实现表格服务端分页、模糊查询
前言 基于SpringMVC+Bootstrap+DataTables实现数据表格服务端分页.模糊查询(非DataTables Search),页面异步刷新. 说明:sp:message标签是使用了S ...
- Inter-partition communication in multi-core processor
A multi-core processor includes logical partitions that have respective processor cores, memory area ...
- android-async-http二次封装和调用
Android android-async-http二次封装和调用 在开发过程中,网络请求这块的使我们常常遇到的一个问题,今天去github 站点上面学习android-async-http,认为还 ...
- V$ASM_OPERATION
- [iOS]iOS获取设备信息经常用法
郝萌主倾心贡献.尊重作者的劳动成果.请勿转载. 假设文章对您有所帮助.欢迎给作者捐赠.支持郝萌主.捐赠数额任意,重在心意^_^ 我要捐赠: 点击捐赠 Cocos2d-X源代码下载:点我传送 游戏官方下 ...