BZOJ 2243 染色 | 树链剖分模板题进阶版

这道题呢就是个带区间修改的树链剖分

如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了。

这道题要注意的是,无论是线段树上还是原树上,把两个区间的信息合并的时候,要注意中间相邻两个颜色是否相同。

这代码好长啊啊啊啊

幸好一次过了不然我估计永远也De不出来

  1. #include <cstdio>
  2. #include <cstring>
  3. #include <algorithm>
  4. using namespace std;
  5. typedef long long ll;
  6. #define space putchar(' ')
  7. #define enter putchar('\n')
  8. template <class T>
  9. void read(T &x){
  10. char c;
  11. bool op = 0;
  12. while(c = getchar(), c < '0' || c > '9')
  13. if(c == '-') op = 1;
  14. x = c - '0';
  15. while(c = getchar(), c >= '0' && c <= '9')
  16. x = x * 10 + c - '0';
  17. if(op) x = -x;
  18. }
  19. template <class T>
  20. void write(T x){
  21. if(x < 0) putchar('-'), x = -x;
  22. if(x >= 10) write(x / 10);
  23. putchar('0' + x % 10);
  24. }
  25. char readchar(){
  26. char c;
  27. while(c = getchar(), c < 'A' || c > 'Z');
  28. return c;
  29. }
  30. const int N = 100005;
  31. int n, m;
  32. int ecnt, adj[N], nxt[2*N], go[2*N];
  33. int tot, pos[N], idx[N], fa[N], son[N], sze[N], top[N], dep[N], val[N];
  34. int le[4*N], ri[4*N], data[4*N], lazy[4*N];
  35. void add(int u, int v){
  36. go[++ecnt] = v;
  37. nxt[ecnt] = adj[u];
  38. adj[u] = ecnt;
  39. }
  40. void pushup(int k){
  41. data[k] = data[k << 1] + data[k << 1 | 1];
  42. if(ri[k << 1] == le[k << 1 | 1]) data[k]--;
  43. le[k] = le[k << 1], ri[k] = ri[k << 1 | 1];
  44. }
  45. void pushdown(int k){
  46. if(lazy[k] == -1) return;
  47. lazy[k << 1] = lazy[k << 1 | 1] = lazy[k];
  48. le[k << 1] = le[k << 1 | 1] = lazy[k];
  49. ri[k << 1] = ri[k << 1 | 1] = lazy[k];
  50. data[k << 1] = data[k << 1 | 1] = 1;
  51. lazy[k] = -1;
  52. }
  53. void build(int k, int l, int r){
  54. lazy[k] = -1;
  55. if(l == r) return (void)(data[k] = 1, le[k] = ri[k] = val[idx[l]]);
  56. int mid = (l + r) >> 1;
  57. build(k << 1, l, mid);
  58. build(k << 1 | 1, mid + 1, r);
  59. pushup(k);
  60. }
  61. void change(int k, int l, int r, int ql, int qr, int x){
  62. if(ql <= l && qr >= r) return (void)(data[k] = 1, le[k] = ri[k] = lazy[k] = x);
  63. pushdown(k);
  64. int mid = (l + r) >> 1;
  65. if(ql <= mid) change(k << 1, l, mid, ql, qr, x);
  66. if(qr > mid) change(k << 1 | 1, mid + 1, r, ql, qr, x);
  67. pushup(k);
  68. }
  69. int query(int k, int l, int r, int ql, int qr){
  70. if(ql <= l && qr >= r) return data[k];
  71. pushdown(k);
  72. int mid = (l + r) >> 1;
  73. if(qr <= mid) return query(k << 1, l, mid, ql, qr);
  74. if(ql > mid) return query(k << 1 | 1, mid + 1, r, ql, qr);
  75. return query(k << 1, l, mid, ql, qr) + query(k << 1 | 1, mid + 1, r, ql, qr) - (ri[k << 1] == le[k << 1 | 1]);
  76. }
  77. int getcol(int k, int l, int r, int p){
  78. if(lazy[k] != -1) return lazy[k];
  79. if(l == r) return le[k];
  80. int mid = (l + r) >> 1;
  81. if(pos[p] <= mid) return getcol(k << 1, l, mid, p);
  82. else return getcol(k << 1 | 1, mid + 1, r, p);
  83. }
  84. void path_change(int u, int v, int x){
  85. if(top[u] == top[v]){
  86. if(pos[u] > pos[v]) swap(u, v);
  87. change(1, 1, n, pos[u], pos[v], x);
  88. return;
  89. }
  90. if(dep[top[u]] > dep[top[v]]) swap(u, v);
  91. change(1, 1, n, pos[top[v]], pos[v], x);
  92. path_change(u, fa[top[v]], x);
  93. }
  94. int path_query(int u, int v){
  95. if(top[u] == top[v]){
  96. if(pos[u] > pos[v]) swap(u, v);
  97. return query(1, 1, n, pos[u], pos[v]);
  98. }
  99. if(dep[top[u]] > dep[top[v]]) swap(u, v);
  100. int same = (getcol(1, 1, n, top[v]) == getcol(1, 1, n, fa[top[v]]));
  101. return path_query(fa[top[v]], u) + query(1, 1, n, pos[top[v]], pos[v]) - same;
  102. }
  103. void init(){
  104. static int que[N], qr;
  105. que[qr = 1] = 1;
  106. for(int ql = 1, u; ql <= qr; ql++){
  107. u = que[ql], sze[u] = 1;
  108. for(int e = adj[u], v; e; e = nxt[e])
  109. if(v = go[e], v != fa[u])
  110. fa[v] = u, dep[v] = dep[u] + 1, que[++qr] = v;
  111. }
  112. for(int ql = qr, u; ql; ql--){
  113. u = que[ql];
  114. sze[fa[u]] += sze[u];
  115. if(sze[u] >= sze[son[fa[u]]]) son[fa[u]] = u;
  116. }
  117. for(int ql = 1, u; ql <= qr; ql++)
  118. if(!top[u = que[ql]])
  119. for(int v = u; v; v = son[v])
  120. idx[++tot] = v, pos[v] = tot, top[v] = u;
  121. build(1, 1, n);
  122. }
  123. int main(){
  124. read(n), read(m);
  125. for(int i = 1; i <= n; i++)
  126. read(val[i]);
  127. for(int i = 1, u, v; i < n; i++)
  128. read(u), read(v), add(u, v), add(v, u);
  129. init();
  130. char op;
  131. int a, b, c;
  132. while(m--){
  133. op = readchar(), read(a), read(b);
  134. if(op == 'Q') write(path_query(a, b)), enter;
  135. else read(c), path_change(a, b, c);
  136. }
  137. return 0;
  138. }

BZOJ 2243 染色 | 树链剖分模板题进阶版的更多相关文章

  1. BZOJ 2243 染色 树链剖分

    题意: 给出一棵树,每个顶点上有个颜色\(c_i\). 有两种操作: C a b c 将\(a \to b\)的路径所有顶点上的颜色变为c Q a b 查询\(a \to b\)的路径上的颜色段数,连 ...

  2. BZOJ - 2243 染色 (树链剖分+线段树+区间合并)

    题目链接 线段树维护区间连续段个数即可.设lc为区间左端点颜色,rc为区间右端点颜色,则合并两区间的时候,如果左区间右端点和右区间左端点颜色相同,则连续段个数-1. 在树链上的区间合并可以定义一个结构 ...

  3. BZOJ 1036 树的统计Count 树链剖分模板题

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1036 题目大意: 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将 ...

  4. bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题

    [ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...

  5. hysbz 2243 染色(树链剖分)

    题目链接:hysbz 2243 染色 题目大意:略. 解题思路:树链剖分+线段树的区间合并,可是区间合并比較简单,节点仅仅要记录左右端点的颜色就可以. #include <cstdio> ...

  6. 洛谷 P3384 树链剖分(模板题)

    题目描述 如题,已知一棵包含N个结点的树(连通且无环),每个节点上包含一个数值,需要支持以下操作: 操作1: 格式: 1 x y z 表示将树从x到y结点最短路径上所有节点的值都加上z 操作2: 格式 ...

  7. 洛谷 P2146 [NOI2015]软件包管理器 (树链剖分模板题)

    题目描述 Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖(即下载安装这个 ...

  8. spoj - Grass Planting(树链剖分模板题)

    Grass Planting 题意 给出一棵树,树有边权.每次给出节点 (u, v) ,有两种操作:1. 把 u 到 v 路径上所有边的权值加 1.2. 查询 u 到 v 的权值之和. 分析 如果这些 ...

  9. P3833 [SHOI2012]魔法树 (树链剖分模板题)

    题目链接:https://www.luogu.org/problem/P3833 题目大意:有一颗含有n个节点的树,初始时每个节点的值为0,有以下两种操作: 1.Add u v d表示将点u和v之间的 ...

随机推荐

  1. python通讯录系统

    ---恢复内容开始--- 对于一般的通讯录系统,主要有两个参数:姓名和电话号码,所以可以利用python编程里面的字典来进行建立之间的所属关系, 可以利用以下代码简单实现: print('|--- 欢 ...

  2. 用UGUI制作可根据手指位置自动定位的隐形遥杆

    之前写过遥杆怎么做,这里依然用的是之前的方法,就不介绍了. 之前玩过<蜡烛人>,发现手游版的<蜡烛人>的遥杆是看不见的,手指直接在屏幕左边滑动人物就可以移动,可能是为了增强沉浸 ...

  3. CSS基础范例

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  4. hostname命令详解

    基础命令学习目录首页 原文链接:https://idc.wanyunshuju.com/cym/68.html Linux操作系统的hostname是一个kernel变量,可以通过hostname命令 ...

  5. uptime命令详解

    基础命令学习目录首页 users个数和窗口数一致 原文链接:https://www.cnblogs.com/ultranms/p/9253217.html uptime 另外还有一个参数 -V(大写) ...

  6. 定时任务crone表达式demo

    1. cron表达式格式: {秒数} {分钟} {小时} {日期} {月份} {星期} {年份(可为空)} 2. cron表达式各占位符解释: {秒数} ==> 允许值范围: 0~59 ,不允许 ...

  7. ModelAndView在页面上取值时value="{contextConfigLocation=<NONE>}"

    后台: mv.getModel().put("initParam", 1); 页面: <input type="hidden" id="init ...

  8. ListView高效分页

    使用控件自带的分页功能时,会先将所查询的数据全部加载出来,若数据量较大,则造成浏览器端等待时间过长. 然而在庞大的数据量,用户所需要的不过是那么几条,甚至只要其中的一条数据,所以,为了减少开销,每次只 ...

  9. java 框架 面试 2

    1.谈谈你对Struts的理解. 答: 1. struts是一个按MVC模式设计的Web层框架,其实它就是一个大大的servlet,这个Servlet名为ActionServlet,或是ActionS ...

  10. linshi18

    #include<iostream> using namespace std; int n,m,k; #define max 100 char mmap[max][max]; int mm ...