题目链接:https://www.luogu.org/problemnew/show/P3979

除了换根操作都是裸的树剖

所以换根时考虑:

1.我查询的根等于换的根:无影响

2.我查询的根是换的根的子树:无影响

3.我查询的根是换的根的祖先:查询 除换的根及其往上直到为要查询的根的第一层儿子的祖先(设为S)的子树 以外的所有节点 (此时满足seg[S] <= seg[root] <= seg[S]+size[S]-1)

求LCA 查询1到seg[S]-1 和 seg[S]+size[S]到n 正好避开子树seg[S] 到 seg[S]+size[S]-1

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <iostream>
  4. #include <algorithm>
  5. #define ll long long
  6. #define lson left, mid, rt<<1
  7. #define rson mid + 1, right, rt<<1|1
  8. using namespace std;
  9. const ll maxn = 200000 + 10;
  10. ll tree[maxn<<2], lazy[maxn<<2];
  11. ll seg[maxn], fa[maxn], size[maxn], rev[maxn], son[maxn], deep[maxn], top[maxn], num;
  12. ll node[maxn], res, n, m, root, r;
  13. struct edge{
  14. ll from, to, next;
  15. }e[maxn<<2];
  16. ll head[maxn], cnt;
  17. void add(ll u, ll v)
  18. {
  19. e[++cnt].from = u;
  20. e[cnt].next = head[u];
  21. e[cnt].to = v;
  22. head[u] = cnt;
  23. }
  24. //--------------------------segment_tree
  25. void PushUP(ll rt)
  26. {
  27. tree[rt] = min(tree[rt<<1], tree[rt<<1|1]);
  28. }
  29. void build(ll left, ll right, ll rt)
  30. {
  31. if(left == right)
  32. {
  33. tree[rt] = rev[left];
  34. return;
  35. }
  36. ll mid = (left + right)>>1;
  37. build(lson);
  38. build(rson);
  39. PushUP(rt);
  40. }
  41. void PushDOWN(ll rt)
  42. {
  43. lazy[rt<<1] = lazy[rt];
  44. lazy[rt<<1|1] = lazy[rt];
  45. tree[rt<<1] = lazy[rt];
  46. tree[rt<<1|1] = lazy[rt];
  47. lazy[rt] = 0;
  48. }
  49. void update(ll l, ll r, ll add, ll left, ll right, ll rt)
  50. {
  51. if(l <= left && r >= right)
  52. {
  53. tree[rt] = add;
  54. lazy[rt] = add;
  55. return;
  56. }
  57. ll mid = (left + right)>>1;
  58. if(lazy[rt]) PushDOWN(rt);
  59. if(l <= mid) update(l, r, add, lson);
  60. if(r > mid) update(l, r, add, rson);
  61. PushUP(rt);
  62. }
  63. ll query(ll l, ll r, ll left, ll right, ll rt)
  64. {
  65. ll res = 0x7fffffff;
  66. if(l <= left && r >= right)
  67. {
  68. return tree[rt];
  69. }
  70. ll mid = (left + right)>>1;
  71. if(lazy[rt]) PushDOWN(rt);
  72. if(l <= mid) res = min(res, query(l, r, lson));
  73. if(r > mid) res = min(res, query(l, r, rson));
  74. return res;
  75. }
  76. //------------------------
  77. void dfs1(ll u, ll f, ll d)
  78. {
  79. ll maxson = -1;
  80. size[u] = 1;
  81. deep[u] = d;
  82. fa[u] = f;
  83. for(ll i = head[u]; i != -1; i = e[i].next)
  84. {
  85. ll v = e[i].to;
  86. if(f != v)
  87. {
  88. dfs1(v, u, d+1);
  89. size[u] += size[v];
  90. if(size[v] > maxson) son[u] = v, maxson = size[v];
  91. }
  92. }
  93. }
  94. void dfs2(ll u, ll t)
  95. {
  96. seg[u] = ++num;
  97. rev[num] = node[u];
  98. top[u] = t;
  99. if(!son[u]) return;
  100. dfs2(son[u], t);
  101. for(ll i = head[u]; i != -1; i = e[i].next)
  102. {
  103. ll v = e[i].to;
  104. if(v == son[u] || v == fa[u]) continue;
  105. dfs2(v,v);
  106. }
  107. }
  108. /*ll qRange(ll x, ll y)
  109. {
  110. ll ans = 0;
  111. while(top[x] != top[y])
  112. {
  113. if(deep[top[x]] < deep[top[y]]) swap(x, y);
  114. res = 0;
  115. res = query(seg[top[x]], seg[x], 1, n, 1);
  116. ans = (ans + res);
  117. x = fa[top[x]];
  118. }
  119. if(deep[x] > deep[y]) swap(x, y);
  120. res = 0;
  121. res = query(seg[x], seg[y], 1, n, 1);
  122. ans = (ans + res);
  123. return ans;
  124. }*/
  125. void updRange(ll x, ll y, ll k)
  126. {
  127. while(top[x] != top[y])
  128. {
  129. if(deep[top[x]] < deep[top[y]]) swap(x, y);
  130. update(seg[top[x]], seg[x], k, 1, n, 1);
  131. x = fa[top[x]];
  132. }
  133. if(deep[x] > deep[y]) swap(x, y);
  134. update(seg[x], seg[y], k, 1, n, 1);
  135. }
  136. ll LCA(int x, int y)
  137. {
  138. while(top[x] != top[y])
  139. {
  140. if(deep[top[x]] < deep[top[y]]) swap(x, y);
  141. x = fa[top[x]];
  142. }
  143. return deep[x] < deep[y] ? x : y;
  144. }
  145. ll qSon(ll x)
  146. {
  147. if(x == r)
  148. return query(1, n, 1, n, 1);
  149. int p = LCA(r, x);
  150. if(p != x)
  151. return query(seg[x], seg[x]+size[x]-1, 1, n, 1);
  152. int S;
  153. for(int i = head[x]; i != -1; i = e[i].next)
  154. {
  155. int v = e[i].to;
  156. if(seg[v] <= seg[r] && seg[r] <= seg[v] + size[v] -1 && v != fa[x])
  157. {
  158. S = v; break;
  159. }
  160. }
  161. return min(query(1, seg[S]-1, 1, n, 1), query(seg[S]+size[S], n, 1, n, 1));
  162. }
  163. int main()
  164. {
  165. memset(head, -1, sizeof(head));
  166. scanf("%lld%lld",&n,&m);
  167. for(ll i = 1; i < n; i++)
  168. {
  169. ll u, v;
  170. scanf("%lld%lld",&u,&v);
  171. add(u, v), add(v, u);
  172. }
  173. for(ll i = 1; i <= n; i++)
  174. scanf("%lld",&node[i]);
  175. scanf("%lld",&root);
  176. dfs1(root, 0, 1);
  177. dfs2(root, root);
  178. build(1,n,1);
  179. r = root;
  180. for(ll i = 1; i <= m; i++)
  181. {
  182. ll opt, x, y, z;
  183. scanf("%lld",&opt);
  184. if(opt == 1)
  185. {
  186. scanf("%lld",&x);
  187. r = x;
  188. }
  189. if(opt == 2)
  190. {
  191. scanf("%lld%lld%lld",&x,&y,&z);
  192. updRange(x, y, z);
  193. }
  194. if(opt == 3)
  195. {
  196. scanf("%lld",&x);
  197. printf("%lld\n",qSon(x));
  198. }
  199. }
  200. return 0;
  201. }

【luogu P3979 遥远的国度】 题解的更多相关文章

  1. P3979 遥远的国度

    P3979 遥远的国度 思路 一开始我用这个函数得到左端点 int get_l(int x,int y) { if(top[x]==top[y]) return son[x]; int last=to ...

  2. P3979 遥远的国度 树剖

    P3979 遥远的国度 树剖 题面 需要想一下的树剖题,对于询问三需要处理换跟后的情况.我们以1为树根跑一遍剖分,对于换跟进行分类讨论,算出实际答案.讨论有三种情况: (以1为树根的树上) 跟在询问节 ...

  3. 【Luogu】P3979遥远的国度(树链剖分)

    题目链接 不会换根从暑假开始就困扰我了……拖到现在…… 会了还是很激动的. 换根操作事实上不需要(也不能)改树剖本来的dfs序……只是在query上动动手脚…… 设全树的集合为G,以root为根,u在 ...

  4. 【树链剖分换根】P3979 遥远的国度

    Description zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要zcw ...

  5. 【luogu P3946 ことりのおやつ】 题解

    题目链接:https://www.luogu.org/problemnew/show/P3946 交好几遍是因为虽然能过一直有提醒..强迫症qwq #include <bits/stdc++.h ...

  6. [洛谷P3979]遥远的国度

    题目大意:有一棵$n$个点的树,每个点有一个点权,有三种操作: $1\;x:$把根变成$x$ $2\;u\;v\;x:$把路径$u->v$上的点权改为$x$ $3\;x:$询问以$x$为根的子树 ...

  7. Luogu P2210 Haywire 题解

    其实这题吧...有一种玄学解法 这题的要求的就是一个最小化的顺序 那么,我们就不进想到了一种显然的写法 就是random_shuffle 什么?这不是乱搞的非正解吗 然而,正如一句话说的好 一个算法, ...

  8. [Luogu P4178]Tree 题解(点分治+平衡树)

    题目大意 给定一棵树,边带权,问有多少点对满足二者间距离$\leq K$,$n \leq 40000$. 题解 点分治专题首杀!$Jackpot!$ (本来看着题意比较简单想捡个软柿子捏,结果手断了… ...

  9. [火星补锅] 水题大战Vol.2 T1 && luogu P1904 天际线 题解 (线段树)

    前言: 当时考场上并没有想出来...后来也是看了题解才明白 解析: 大家(除了我)都知道,奇点和偶点会成对出现,而出现的前提就是建筑的高度突然发生变化.(这个性质挺重要的,我之前没看出来) 所以就可以 ...

随机推荐

  1. 第十六章:自定义push notification sound

    前面一节已经讲过如何在ionic中集成jpush,这样我们的hybrid app在部署到ios或者android上面的时候,就可以接收通知了.如果不满足系统自带的声音,可以通过一些方式来播放自定义的通 ...

  2. 在Spark shell中基于HDFS文件系统进行wordcount交互式分析

    Spark是一个分布式内存计算框架,可部署在YARN或者MESOS管理的分布式系统中(Fully Distributed),也可以以Pseudo Distributed方式部署在单个机器上面,还可以以 ...

  3. vi编辑器备忘录

    1. 基本操作 G 移动到文件最后一行 nG 移动到n行 gg 移动到第一行 N[Enter] 向下移动n行 /word 向下寻找 ?word 向上寻找 n 查找下一个 N 查找上一个 0或者Home ...

  4. 【程序员技术练级】学习一门脚本语言 python(一)文件处理

    现在工作上主要用的语言是java,java在企业级的应用上能够发挥很好的用途,但有时候要做一个小功能时,比如批量更新文件,抓取网页等,这时候用java就显得太笨重了.因此就学习了python这门脚本语 ...

  5. Android中用Java代码实现zip文件解压缩

    如果需要下载的文件有很多是中文名的,解压时有中文名的文件出现乱码,试了很多方法不能解决问题.据说有一个Java插件包,用这个插件包可以解决中文名乱码的问题,但不知解压的文件是否要用它提供的类压缩后的文 ...

  6. Visual Studio 创建封装自己的代码段(C#)

    https://www.cnblogs.com/awaTangjay/p/6644952.html 1.打开vs2012--工具--代码段管理器 2.进入代码管理器之后,语言选择Visual C#,然 ...

  7. Wireshark使用技巧

    Wireshark使用技巧 在分析网络时,包应该尽量的小,只要能定位问题即可. 1. 只抓包头,在wireshark中可以设置抓包大小. 如果使用tcpdump命令: [root@server_1 / ...

  8. JVM 类加载全过程

    类加载机制 - JVM把class文件加载到内存中 并对数据进行 校验,解析,初始化,最终形成JVM可以直接使用的java类型的过程 详细过程  加载→ 验证→ 准备→ 解析 → 初始化→ 使用 → ...

  9. BJFU 1549 ——Candy——————【想法题】

    Candy 时间限制(C/C++):1000MS/3000MS          运行内存限制:65536KByte总提交:40            测试通过:20 描述 There are N c ...

  10. 从零开始写C# MVC框架之--- 配置log4日志

    在框架中配置日志分2步,一个是在帮助项目Zy.Utilities--Zy.Utility.Core中新建log类,封装写入日志方法,还需要在Zy.Utility.Core添加 log4net 的引用 ...