C. Propagating tree

Time Limit: 20 Sec

Memory Limit: 256 MB

题目连接

http://codeforces.com/contest/383/problem/C

Description

Iahub likes trees very much. Recently he discovered an interesting tree named propagating tree. The tree consists of n nodes numbered from 1 to n, each node i having an initial value ai. The root of the tree is node 1.

This tree has a special property: when a value val is added to a value of node i, the value -val is added to values of all the children of node i. Note that when you add value -val to a child of node i, you also add -(-val) to all children of the child of node i and so on. Look an example explanation to understand better how it works.

This tree supports two types of queries:

​ "1 x val" — val is added to the value of node x;

​ "2 x" — print the current value of node x.

In order to help Iahub understand the tree better, you must answer m queries of the preceding type.

Input

The first line contains two integers n and m (1 ≤ n, m ≤ 200000). The second line contains n integers a1, a2, ..., an (1 ≤ ai ≤ 1000). Each of the next n–1 lines contains two integers vi and ui (1 ≤ vi, ui ≤ n), meaning that there is an edge between nodes vi and ui.

Each of the next m lines contains a query in the format described above. It is guaranteed that the following constraints hold for all queries: 1 ≤ x ≤ n, 1 ≤ val ≤ 1000.

Output

For each query of type two (print the value of node x) you must print the answer to the query on a separate line. The queries must be answered in the order given in the input.

Sample Input

5 5

1 2 1 1 2

1 2

1 3

2 4

2 5

1 2 3

1 1 2

2 1

2 2

2 4

Sample Output

3

3

0

HINT

题意

给出一颗有n个节点并一1为根节点的树,每个节点有它的权值,现在进行m次操作,操作分为添加和查询,当一个节点的权值添加val,则它的孩子节点的权值要添加-b。

题解:

方法一:dfs序+树状数组

学过或者没学过树链剖分的小朋友应该知道dfs序可以使得任意节点的子树变成一段连续的区间,这样就可以进行区间修改了。

但是这道题对子树层数的不同,操作也不一样,确实很麻烦呀。

我也是一脸懵逼,看完题解才发现原来如此神奇!!

其实我们可以把树分层,分成奇数层和偶数层(根据到根节点的距离),然后建两个树状数组,如果是对奇数层的点进行操作那么就给第一个树状数组相应的区间加上val,如果是偶数层就更新第二个树状数组。

这样对于某个节点i(假设是奇数层),那么此刻权值就是树状数组1-树状数组2。

感觉这种分层的思想很是神奇。

方法二:dfs序+线段树

第二天遇到另外一题 https://www.cnblogs.com/mmmqqdd/p/10844365.html ,突然发现这道题就是那题的简化版,大家可以先看看那题,

因为这道题用线段树实在是有点杀鸡用牛刀了,但是思想和不错哟。

我在这也简单的讲一下做法吧:

就是线段树每个节点都标记是奇数层还是偶数层,奇数层为-1,偶数层为1,然后如果处理奇数层就区间-val,偶数层就+val,其他的和线段树一样,最后求答案时,只要把最后的值×1或-1(奇数层-1,偶数层1)。

这个题目完全可以问一棵子数的权值和,这样像一棵正常的线段树了。

代码1--树状数组:

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define N 400050
int n,m,cnt,w[N],a[N],c[2][N];
int dp[N],rk[N],kth[N],size[N];
int tot,last[N];
struct Edge{int from,to,s;}edges[N<<1];
template<typename T>void read(T&x)
{
ll k=0; char c=getchar();
x=0;
while(!isdigit(c)&&c!=EOF)k^=c=='-',c=getchar();
if (c==EOF)exit(0);
while(isdigit(c))x=x*10+c-'0',c=getchar();
x=k?-x:x;
}
void read_char(char &c)
{while(!isalpha(c=getchar())&&c!=EOF);}
void AddEdge(int x,int y)
{
edges[++tot]=Edge{x,y,last[x]};
last[x]=tot;
}
void dfs(int x,int pre)
{
dp[x]=1-dp[pre];
rk[x]=++cnt;
kth[cnt]=x;
size[x]=1;
for(int i=last[x];i;i=edges[i].s)
{
Edge &e=edges[i];
if (e.to==pre)continue;
dfs(e.to,x);
size[x]+=size[e.to];
}
}
void update(int *c,int x,int tt)
{while(x<=n){c[x]+=tt;x+=x&-x;}}
int get_sum(int *c,int x)
{int ans=0;while(x){ans+=c[x];x-=x&-x;}return ans;}
int main()
{
#ifndef ONLINE_JUDGE
freopen("aa.in","r",stdin);
#endif
read(n);read(m);
for(int i=1;i<=n;i++)read(w[i]);
for(int i=1;i<=n-1;i++)
{
int x,y;
read(x); read(y);
AddEdge(x,y);
AddEdge(y,x);
}
dfs(1,0);
for(int i=1;i<=m;i++)
{
int id,x,val;
read(id);
if (id==1)
{
read(x); read(val);
update(c[dp[x]],rk[x],val);
update(c[dp[x]],rk[x]+size[x],-val);
}
if (id==2)
{
read(x);
int ans=get_sum(c[dp[x]],rk[x])-get_sum(c[1-dp[x]],rk[x]);
ans+=w[x];
printf("%d\n",ans);
}
} }

代码2--线段树:

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define N 200050
int n,m,w[N];
int tot,last[N];
int cnt,rk[N],kth[N],d[N],size[N];
struct Tree{int l,r,c;}tr[N<<2];
struct Edge{int from,to,s;}edges[N<<1];
void AddEdge(int x,int y)
{
edges[++tot]=Edge{x,y,last[x]};
last[x]=tot;
}
template<typename T>void read(T&x)
{
ll k=0; char c=getchar();
x=0;
while(!isdigit(c)&&c!=EOF)k^=c=='-',c=getchar();
if (c==EOF)exit(0);
while(isdigit(c))x=x*10+c-'0',c=getchar();
x=k?-x:x;
}
void read_char(char &c)
{while(!isalpha(c=getchar())&&c!=EOF);}
void dfs(int x,int pre)
{
rk[x]=++cnt;
kth[cnt]=x;
d[x]=-d[pre];
size[x]=1;
for(int i=last[x];i;i=edges[i].s)
{
Edge &e=edges[i];
if (e.to==pre)continue;
dfs(e.to,x);
size[x]+=size[e.to];
}
}
void push_down(int x)
{
Tree &a=tr[x<<1],&b=tr[x<<1|1];
a.c+=tr[x].c;
b.c+=tr[x].c;
tr[x].c=0;
}
void bt(int x,int l,int r)
{
tr[x].l=l; tr[x].r=r;
if (l==r) return;
int mid=(l+r)>>1;
bt(x<<1,l,mid);
bt(x<<1|1,mid+1,r);
}
void update(int x,int l,int r,int tt)
{
if (l<=tr[x].l&&tr[x].r<=r)
{tr[x].c+=tt;return;}
int mid=(tr[x].l+tr[x].r)>>1;
if (l<=mid)update(x<<1,l,r,tt);
if (mid<r)update(x<<1|1,l,r,tt);
}
int query(int x,int p)
{
if (p<=tr[x].l&&tr[x].r<=p)
return tr[x].c*d[kth[tr[x].l]];
int mid=(tr[x].l+tr[x].r)>>1;
push_down(x);
if (p<=mid)return query(x<<1,p);
else return query(x<<1|1,p);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("aa.in","r",stdin);
#endif
read(n); read(m);
for(int i=1;i<=n;i++)read(w[i]);
for(int i=1;i<=n-1;i++)
{
int x,y;
read(x); read(y);
AddEdge(x,y);
AddEdge(y,x);
}
d[0]=1;
dfs(1,0);
bt(1,1,n);
for(int i=1;i<=m;i++)
{
int id,x,tt;
read(id); read(x);
if (id==1)
{
read(tt);
update(1,rk[x],rk[x]+size[x]-1,tt*d[x]);
}
if (id==2)
printf("%d\n",w[x]+query(1,rk[x]));
}
}

最新文章

  1. Linux下显示IP地理位置信息的小工具-nali
  2. unity htc vive, ugui for vr
  3. windows phone 8 开发系列(一)环境搭建
  4. template和templateUrl区别与联系
  5. VS2010如何调试IIS上的网站
  6. 【CSS】Intermediate1:Class and ID Selectors
  7. Spring各种传播特性源码实现的概览
  8. js中邦定事件与解绑支持匿名函数
  9. 在WHERE子句中引用取别名的列
  10. cocos2dx进阶学习之CCApplication
  11. Lucene.Net 2.3.1开发介绍 —— 二、分词(三)
  12. git初步用法
  13. wpf研究之道-grid控件
  14. emqtt 试用(一)安装和测试
  15. SpringMVC(六):@RequestMapping下使用@RequestHeader绑定请求报头的属性值、@CookieValue绑定请求中的Cookie值
  16. js的splice方法
  17. applium安装过程中遇到的问题及解决方法。
  18. 框架: Struts2 讲解 1
  19. Linux 组管理、权限
  20. selenium之生成html测试报告--testng.xsl

热门文章

  1. P2699 【数学1】小浩的幂次运算
  2. 决策树算法的Python实现—基于金融场景实操
  3. Java并发之同步工具类
  4. Python控制台输出时刷新当前行内容而不是输出新行
  5. Leetcode题目101.对称二叉树(简单)
  6. Ajax传递复杂对象报415
  7. Winform运行外部控制台程序,并在程序结束后执行其他动作
  8. UIGestureRecongnizer 手势拦截 对于特殊需求很有用
  9. cat中文正常vi中文乱码
  10. 关于JavaWeb面试