题目

Link

分析

典型的树链剖分题,

树链剖分学习资料

Code

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int maxn = 30000 + 131;
  4. struct Edge {
  5. int Next;
  6. int To;
  7. }edge[maxn<<1];
  8. int Head[maxn], tot, n;
  9. ///以下是重链数据定义
  10. int top[maxn]; //重链的顶点
  11. int deep[maxn]; //树上节点的深度
  12. int Pre[maxn]; //父节点
  13. int size[maxn]; //子树节点大小
  14. int son[maxn]; //重链中节点的子节点
  15. ///以下有关离散到线段树数据定义
  16. int t_s[maxn]; //树上的点离散到线段树
  17. int s_t[maxn]; //线段树映射回树上的点。
  18. int pos;
  19. //题目数据
  20. int w[maxn];
  21. void INIT() {
  22. tot = pos = 0;
  23. memset(son, -1, sizeof(son));
  24. memset(Head,-1, sizeof(Head));
  25. }
  26. ////
  27. void Addedge(int from, int to) {
  28. edge[tot].To = to;
  29. edge[tot].Next = Head[from];
  30. Head[from] = tot++;
  31. }
  32. void Getlist(int root, int pre, int d) { //获得重链
  33. deep[root] = d;
  34. Pre[root] = pre;
  35. size[root] = 1;
  36. for(int i = Head[root]; ~i; i = edge[i].Next) {
  37. int v = edge[i].To;
  38. if(v != pre) {
  39. Getlist(v, root, d+1);
  40. size[root] += size[v]; //累加size
  41. if(son[root] == -1 || size[son[root]] < size[v])
  42. son[root] = v; //更新重链子节点
  43. }
  44. }
  45. }
  46. ////离散点到线段树上
  47. void Lisan_TtoS(int u, int root) {
  48. top[u] = root;
  49. t_s[u] = ++pos;
  50. s_t[t_s[u]] = u;
  51. if(son[u] == -1) return ;
  52. Lisan_TtoS(son[u], root);
  53. for(int i = Head[u]; ~i; i = edge[i].Next) {
  54. int v = edge[i].To;
  55. if(v != son[u] && v != Pre[u])
  56. Lisan_TtoS(v,v); //新的重链开始.
  57. }
  58. }
  59. ////线段树
  60. int Sum[maxn << 2];
  61. #define lson l, m, rt<<1
  62. #define rson m+1, r, rt<<1|1
  63. void PushUp(int rt) {
  64. Sum[rt] = Sum[rt<<1] + Sum[rt<<1|1];
  65. }
  66. void Build(int l, int r, int rt) {
  67. if(l == r) {
  68. Sum[rt] = w[s_t[l]];
  69. /*cout << "This is tree idx:" << s_t[l] \
  70. << " values:" << w[s_t[l]] << endl;*/
  71. return ;
  72. }
  73. int m = (l + r) >> 1;
  74. Build(lson);
  75. Build(rson);
  76. PushUp(rt);
  77. }
  78. void Update(int pos, int val, int l, int r, int rt) {
  79. if(l == r) {
  80. Sum[rt] = val;
  81. //cout << "This is the tree id: " << s_t[l] << endl;
  82. return ;
  83. }
  84. int m = (l + r) >> 1;
  85. if(pos <= m) Update(pos, val, lson);
  86. else Update(pos, val, rson);
  87. PushUp(rt);
  88. }
  89. int Query(int L, int R, int l, int r, int rt) {
  90. //cout << "seg l : " << l << " r : " << r ;
  91. //cout << " Find L:" << L << " R: " << R << endl;
  92. if(L <= l && r <= R) {
  93. //cout << "Had add : " << Sum[rt] << endl;
  94. return Sum[rt];
  95. }
  96. int m = (l + r) >> 1;
  97. int ret = 0;
  98. if(L <= m) ret += Query(L, R, lson);
  99. if(R > m) ret += Query(L, R, rson);
  100. return ret;
  101. }
  102. ////查询(u,v)
  103. int Find(int u, int v) { /// u to v
  104. int fa_u = top[u]; /// u总是更深的点.
  105. int fa_v = top[v];
  106. int ret = 0;
  107. while(fa_u != fa_v) {
  108. if(deep[fa_u] < deep[fa_v]) {
  109. swap(u, v);
  110. swap(fa_v,fa_u);
  111. }
  112. //cout << "This is the list :" << fa_u << "->" << u << endl;
  113. //cout << "This is the Segm :" << t_s[fa_u] << "->" << t_s[u] << endl;
  114. ret += Query(t_s[fa_u], t_s[u], 1, n, 1);
  115. u = Pre[fa_u];
  116. fa_u = top[u];
  117. }
  118. // 点, 所以会有两点重合的情况。
  119. // 边的处理, 可以理解 i 点 to j 点的边 v 就是 j 点的权值
  120. // root 处理为最小只即可 or (0) or 各种适合值。
  121. if(deep[u] > deep[v]) swap(u, v);
  122. ret += Query(t_s[u], t_s[v], 1, n, 1);
  123. return ret;
  124. }
  125. int main() {
  126. int T;
  127. scanf("%d",&T);
  128. for(int kase = 1; kase <= T; ++kase) {
  129. scanf("%d",&n);
  130. INIT();
  131. for(int i = 1; i <= n; ++i)
  132. scanf("%d",w+i);
  133. int u, v;
  134. for(int i = 1; i < n; ++i) {
  135. scanf("%d%d",&u, &v);
  136. u++, v++;
  137. Addedge(u, v);
  138. Addedge(v, u);
  139. }
  140. Getlist(1, -1, 0);
  141. Lisan_TtoS(1,1);
  142. Build(1, n, 1);
  143. printf("Case %d:\n",kase);
  144. int q;
  145. scanf("%d",&q);
  146. for(int i = 0; i < q; ++i) {
  147. int op;
  148. scanf("%d%d%d",&op,&u,&v);
  149. if(op == 1) {
  150. Update(t_s[++u], v, 1, n, 1);
  151. }
  152. else {
  153. u++, v++;
  154. printf("%d\n", Find(u, v));
  155. }
  156. }
  157. }
  158. return 0;
  159. }

LightOJ 1348 (树链剖分 + 线段树(树状数组))的更多相关文章

  1. 【BZOJ-2325】道馆之战 树链剖分 + 线段树

    2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1153  Solved: 421[Submit][Statu ...

  2. 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树

    [BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...

  3. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

  4. POJ3237 (树链剖分+线段树)

    Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...

  5. bzoj4034 (树链剖分+线段树)

    Problem T2 (bzoj4034 HAOI2015) 题目大意 给定一颗树,1为根节点,要求支持三种操作. 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子 ...

  6. HDU4897 (树链剖分+线段树)

    Problem Little Devil I (HDU4897) 题目大意 给定一棵树,每条边的颜色为黑或白,起始时均为白. 支持3种操作: 操作1:将a->b的路径中的所有边的颜色翻转. 操作 ...

  7. Aizu 2450 Do use segment tree 树链剖分+线段树

    Do use segment tree Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.bnuoj.com/v3/problem_show ...

  8. 【POJ3237】Tree(树链剖分+线段树)

    Description You are given a tree with N nodes. The tree’s nodes are numbered 1 through N and its edg ...

  9. HDU 2460 Network(双连通+树链剖分+线段树)

    HDU 2460 Network 题目链接 题意:给定一个无向图,问每次增加一条边,问个图中还剩多少桥 思路:先双连通缩点,然后形成一棵树,每次增加一条边,相当于询问这两点路径上有多少条边,这个用树链 ...

随机推荐

  1. CodeForces 1151F Sonya and Informatics

    题目链接:http://codeforces.com/problemset/problem/1151/F 题目大意: 给定长度为 n 的 01 序列,可以对该序列操作 k 次,每次操作可以交换序列中任 ...

  2. 调用webservice帮助类

    using System;using System.CodeDom;using System.CodeDom.Compiler;using System.Collections.Generic;usi ...

  3. nodejs 搭建本地静态服务器

    1. http-server 参看 https://www.npmjs.com/package/http-server 使用http-server搭建本地静态服务器 全局安装http-server n ...

  4. HBuilderX——编译失败:HBuilderX 安装目录不能包括 ( 等特殊字符

    前言 编译小程序的时候报错,原因其实很简单,安装目录的问题! 解决 我安装到了D:\Program Files (x86),放到D盘的根目录下或者D:\Program Files只要不包含一些特殊字符 ...

  5. 戏说java与web

    slmgr.vbs /ipk NPPR9-FWDCX-D2C8J-H872K-2YT43教育版换回企业版 搜百度网盘  http://www.pansoso.com/ https://m.zhangl ...

  6. 微信小程序onLaunch、onLoad执行生命周期

    原文转载自:微信小程序onLaunch.onLoad执行生命周期 1.需求:先执行App的onLaunch添加验证权限等,再执行Page里的onLoad. 2.问题:还没有等onLaunch执行完成, ...

  7. PHP 消息队列 详解

    前言:之前做过的一些项目中有时候会接触到消息队列,但是对消息队列并没有一个很清楚的认知,本篇文章将会详细分析和归纳一些笔记,以供后续学习. 一.消息对列概念 从本质上说消息对列就是一个队列结构的中间件 ...

  8. python第六天

    深浅拷贝,元祖,字典 ,集合的定义以及基本操作方法 深浅拷贝 # 值拷贝:应用场景最多​值拷贝:先创建一个列表ls = [1, 'abc', [10]] 再定义 ls1 = ls  此处ls1会直接将 ...

  9. es6异步编程

    https://blog.csdn.net/tcy83/article/details/80274772 等一系列文章

  10. Flash设置(各种版本浏览器包括低版本IE)

    涉及到的各种版本flash百度下都能下到的,不再说明. Flash的安装比较麻烦,涉及多种浏览器.多种操作系统支持,安装.设置的地方比较多,以下说明基本涉及大部分安装过程中可能遇到的问题,如果安装或视 ...