题意:题意很简单么,给定n个点,m个询问的无向树(1为根),每个点的权值,有两种操作,

第一种:1 x v,表示把 x 结点加上v,然后把 x 的的子结点加上 -v,再把 x 的子结点的子结点加上 -(-v),依次。。。

第二种:2 x, 表示查询 x 结点的权值。

析:因为这是一棵树,很难维护,所以可以考虑先用 dfs 记录每个结点的开始和结束的时间戳,而位于开始和结束时间戳内的就是就是它的子孙结点,

这样就能维护两棵线段树,一棵是奇数层的, 一棵是偶数层的,每次执行 1 操作就把相应的结点的开始和结束作为一个区间进行更新,然后再执行

相反层的 -v 进行更新。当查询的时候,就直接输出相应层的输出再加原来权值即可。

代码如下:

  1. #pragma comment(linker, "/STACK:1024000000,1024000000")
  2. #include <cstdio>
  3. #include <string>
  4. #include <cstdlib>
  5. #include <cmath>
  6. #include <iostream>
  7. #include <cstring>
  8. #include <set>
  9. #include <queue>
  10. #include <algorithm>
  11. #include <vector>
  12. #include <map>
  13. #include <cctype>
  14. #include <cmath>
  15. #include <stack>
  16. #include <sstream>
  17. #define debug() puts("++++");
  18. #define gcd(a, b) __gcd(a, b)
  19. #define lson l,m,rt<<1
  20. #define rson m+1,r,rt<<1|1
  21. #define freopenr freopen("in.txt", "r", stdin)
  22. #define freopenw freopen("out.txt", "w", stdout)
  23. using namespace std;
  24.  
  25. typedef long long LL;
  26. typedef unsigned long long ULL;
  27. typedef pair<int, int> P;
  28. const int INF = 0x3f3f3f3f;
  29. const LL LNF = 1e16;
  30. const double inf = 0x3f3f3f3f3f3f;
  31. const double PI = acos(-1.0);
  32. const double eps = 1e-8;
  33. const int maxn = 200000 + 10;
  34. const int mod = 1000000007;
  35. const int dr[] = {-1, 0, 1, 0};
  36. const int dc[] = {0, 1, 0, -1};
  37. const char *de[] = {"0000", "0001", "0010", "0011", "0100", "0101", "0110", "0111", "1000", "1001", "1010", "1011", "1100", "1101", "1110", "1111"};
  38. int n, m;
  39. const int mon[] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
  40. const int monn[] = {0, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
  41. inline bool is_in(int r, int c){
  42. return r >= 0 && r < n && c >= 0 && c < m;
  43. }
  44.  
  45. int dfsnum[maxn], in[maxn];
  46. int a[maxn], out[maxn], dep[maxn];
  47. int sum[2][maxn<<2], addv[2][maxn<<2];
  48. vector<int> G[maxn];
  49. int cnt;
  50.  
  51. void dfs(int u, int fa, int d){
  52. dep[u] = d;
  53. in[u] = ++cnt;
  54.  
  55. for(int i = 0; i < G[u].size(); ++i){
  56. int v = G[u][i];
  57. if(v == fa) continue;
  58. dfs(v, u, d+1);
  59. }
  60. out[u] = cnt;
  61. }
  62.  
  63. void push_down(int *sum, int rt, int *addv){
  64. if(addv[rt] == 0) return ;
  65. int l = rt<<1, r = rt<<1|1;
  66. sum[l] += addv[rt];
  67. sum[r] += addv[rt];
  68. addv[l] += addv[rt];
  69. addv[r] += addv[rt];
  70. addv[rt] = 0;
  71. }
  72.  
  73. void update(int *sum, int *addv, int L, int R, int val, int l, int r, int rt){
  74. if(L <= l && r <= R){
  75. addv[rt] += val;
  76. sum[rt] += val;
  77. return ;
  78. }
  79. push_down(sum, rt, addv);
  80. int m = l + r >> 1;
  81. if(L <= m) update(sum, addv, L, R, val, lson);
  82. if(R > m) update(sum, addv, L, R, val, rson);
  83. }
  84.  
  85. int query(int *sum, int *addv, int M, int l, int r, int rt){
  86. if(l == r) return sum[rt];
  87. push_down(sum, rt, addv);
  88. int m = l + r >> 1;
  89. if(M <= m) return query(sum, addv, M, lson);
  90. return query(sum, addv, M, rson);
  91. }
  92.  
  93. int main(){
  94. scanf("%d %d", &n, &m);
  95. for(int i = 1; i <= n; ++i) scanf("%d", a+i);
  96. for(int i = 1; i < n; ++i){
  97. int u, v;
  98. scanf("%d %d", &u, &v);
  99. G[u].push_back(v);
  100. G[v].push_back(u);
  101. }
  102. dfs(1, -1, 1);
  103. while(m--){
  104. int x, c, op;
  105. scanf("%d %d", &op, &x);
  106. int t = dep[x]&1;
  107. if(op == 1){
  108. scanf("%d", &c);
  109. update(sum[t], addv[t], in[x], out[x], c, 1, n, 1);
  110. if(in[x] < out[x])
  111. update(sum[t^1], addv[t^1], in[x]+1, out[x], -c, 1, n, 1);
  112. }
  113. else printf("%d\n", query(sum[t], addv[t], in[x], 1, n, 1) + a[x]);
  114. }
  115. return 0;
  116. }

  

CodeForces 384E Propagating tree (线段树+dfs)的更多相关文章

  1. codeforces 383C Propagating tree 线段树

    http://codeforces.com/problemset/problem/383/C 题目就是说,  给一棵树,将一个节点的值+val, 那么它的子节点都会-val, 子节点的子节点+val. ...

  2. Codeforces 383C Propagating tree, 线段树, 黑白染色思想

    按深度染色,奇深度的点存反权值. #include <bits/stdc++.h> using namespace std; vector <]; ],a[],s[],vis[],i ...

  3. Codeforces 383C . Propagating tree【树阵,dfs】

    标题效果: 有一棵树,有两种操作模式对本树:1:表示为(1 x val),在NOx加在节点上val,然后x每个节点加上儿子- val.给每个儿子一个儿子在一起-(- val),加到没有儿子为止.2:表 ...

  4. CF620E New Year Tree 线段树 dfs序

    luogu链接 题目大意: 有一个节点有颜色的树 操作1.修改子树的颜色 操作2.查询子树颜色的种类 注意,颜色种类小于60种 只有子树的操作,dfs序当然是最好的选择 dfs序列是什么,懒得讲了,自 ...

  5. CF620E New Year Tree 线段树+dfs序+bitset

    线段树维护 dfs 序是显然的. 暴力建 60 个线段树太慢,于是用 bitset 优化就好了 ~ code: #include <bits/stdc++.h> #define M 63 ...

  6. S - Query on a tree HDU - 3804 线段树+dfs序

    S - Query on a tree HDU - 3804   离散化+权值线段树 题目大意:给你一棵树,让你求这棵树上询问的点到根节点直接最大小于等于val的长度. 这个题目和之前写的那个给你一棵 ...

  7. HDU 5692 线段树+dfs序

    Snacks Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Sub ...

  8. Tsinsen A1505. 树(张闻涛) 倍增LCA,可持久化线段树,DFS序

    题目:http://www.tsinsen.com/A1505 A1505. 树(张闻涛) 时间限制:1.0s   内存限制:512.0MB    总提交次数:196   AC次数:65   平均分: ...

  9. 【XSY2534】【BZOJ4817】树点涂色 LCT 倍增 线段树 dfs序

    题目大意 ​ Bob有一棵\(n\)个点的有根树,其中\(1\)号点是根节点.Bob在每个点上涂了颜色,并且每个点上的颜色不同.定义一条路径的权值是:这条路径上的点(包括起点和终点)共有多少种不同的颜 ...

随机推荐

  1. 让camera实现类似cs第一人称视角旋转和位移

    直接把这个脚本挂在摄像机上就可: using System.Collections; using System.Collections.Generic; using UnityEngine; /* * ...

  2. How To Uninstall Software Using The Ubuntu Command Line

    How To Uninstall Software Using The Ubuntu Command Line Uninstall Ubuntu Software Using The Terminal ...

  3. 使用case语句给字体改变颜色

    使用case语句给字体改变颜色 #!/bin/bash color(){ RED_COLOR='\E[1;31m' GREEN_COLOR='\E[1;32m' YELLOW_COLOR='\E[1; ...

  4. Java 8新增的日期、时间格式器

    一 获取DateTimeFormatter对象的三种方式 直接使用静态常量创建DateTimeFormatter格式器 使用代码不同风格的枚举值来创建DateTimeFormatter格式器 根据模式 ...

  5. 2015 浙江省赛 H - May Day Holiday

    H - May Day Holiday As a university advocating self-learning and work-rest balance, Marjar Universit ...

  6. webbrowser和js交互小结

    一.实现WebBrowser内部跳转,阻止默认打开IE 1.引用封装好的WebBrowserLinkSelf.dll实现 public partial class MainWindow : Windo ...

  7. app中使用微信分享注意事项

    1.  在微信公众平台开通一个微信公众号,https://mp.weixin.qq.com 2.  将自己制作好的已签名的app安装到手机上 3.  下载微信开放平台获取应用签名的apk--- gen ...

  8. delphi 蓝牙 TBluetoothLE

    delphi 蓝牙 TBluetoothLE.TBluetoothLEManager BLE http://docwiki.embarcadero.com/RADStudio/Seattle/en/U ...

  9. 部署和调优 2.7 mysql主从配置-1

    MySQL 主从(MySQL Replication),主要用于 MySQL 的时时备份或者读写分离.在配置之前先做一下准备工作,配置两台 mysql 服务器,如果你的机器不能同时跑两台 Linux虚 ...

  10. adb pull 和 adb push

    1. 操作单个文件 通过adb push,则可将文件添加到SD卡中.如果想在push的时候修改文件名称的话,只需要修改push的第二个参数改成完整路径(目录+文件名),如/sdcard/test-0. ...