POJ2763 Housewife Wind 树链剖分 边权

传送门:http://poj.org/problem?id=2763

题意:

n个点的,n-1条边,有边权

修改单边边权

询问 输出 当前节点到 x节点的最短距离,并移动到 x 节点位置

题解:

树链剖分裸题

树链剖分就是将树分割为多条边,然后利用数据结构来维护这些链的一个技巧

  • 重儿子:父亲节点的所有儿子中子树结点数目最多( sizesiz**e 最大)的结点;
  • 轻儿子:父亲节点中除了重儿子以外的儿子;
  • 重边:父亲结点和重儿子连成的边;
  • 轻边:父亲节点和轻儿子连成的边;
  • 重链:由多条重边连接而成的路径;
  • 轻链:由多条轻边连接而成的路径;

1.求出子树大小和每个点的重儿子,处理sz数组,son数组,fa数组和dep数组

2.连接重链,记录dfs序,每个链的顶端节点,处理出rank数组,top数组,id数组

3.维护链上信息

修改边权就查询点的深度大的点,用大的点去存这条边的边权,其余的就和点权的是一样的了

代码:

  1. #include <set>
  2. #include <map>
  3. #include <cmath>
  4. #include <cstdio>
  5. #include <string>
  6. #include <vector>
  7. #include <cstring>
  8. #include <iostream>
  9. #include <algorithm>
  10. using namespace std;
  11. typedef long long LL;
  12. typedef pair<int, int> pii;
  13. typedef unsigned long long uLL;
  14. #define ls rt<<1
  15. #define rs rt<<1|1
  16. #define lson l,mid,rt<<1
  17. #define rson mid+1,r,rt<<1|1
  18. #define bug printf("*********\n")
  19. #define FIN freopen("input.txt","r",stdin);
  20. #define FON freopen("output.txt","w+",stdout);
  21. #define IO ios::sync_with_stdio(false),cin.tie(0)
  22. #define debug1(x) cout<<"["<<#x<<" "<<(x)<<"]\n"
  23. #define debug2(x,y) cout<<"["<<#x<<" "<<(x)<<" "<<#y<<" "<<(y)<<"]\n"
  24. #define debug3(x,y,z) cout<<"["<<#x<<" "<<(x)<<" "<<#y<<" "<<(y)<<" "<<#z<<" "<<z<<"]\n"
  25. const int maxn = 3e5 + 5;
  26. const int INF = 0x3f3f3f3f;
  27. int n, m, Q;
  28. int a[maxn], sz[maxn], dep[maxn], fa[maxn], top[maxn], id[maxn], son[maxn], Rank[maxn];
  29. int sum[maxn << 2], lazy[maxn << 2];
  30. struct EDGE {
  31. int u, v, nt;
  32. } edge[maxn << 1];
  33. int head[maxn], summ, cnt;
  34. void add_edge(int u, int v) {
  35. edge[++summ].u = u; edge[summ].v = v; edge[summ].nt = head[u]; head[u] = summ;
  36. }
  37. void dfs1(int u) {
  38. sz[u] = 1; son[u] = 0;
  39. for (int i = head[u]; ~i; i = edge[i].nt) {
  40. int v = edge[i].v;
  41. if (v != fa[u]) {
  42. fa[v] = u;
  43. dep[v] = dep[u] + 1;
  44. dfs1(v);
  45. sz[u] += sz[v];
  46. if (sz[v] > sz[son[u]]) son[u] = v;
  47. }
  48. }
  49. }
  50. void dfs2(int u, int tp, int x) {
  51. top[u] = tp; id[u] = ++cnt; Rank[cnt] = u;
  52. if (son[u]) dfs2(son[u], tp, 1);
  53. for (int i = head[u]; ~i; i = edge[i].nt) {
  54. int v = edge[i].v;
  55. if (v == son[u] || v == fa[u]) continue;
  56. dfs2(v, v, 2);
  57. }
  58. }
  59. void init() {
  60. memset(head, -1, sizeof(head));
  61. summ = 1; cnt = 0;
  62. for (int i = 1; i <= n; i++) scanf("%d", &a[i]);
  63. for (int i = 1; i <= m; i++) {
  64. int u, v;
  65. scanf("%d %d", &u, &v);
  66. add_edge(u, v); add_edge(v, u);
  67. }
  68. dep[1] = 1; fa[1] = 0;
  69. dfs1(1);
  70. dfs2(1, 1, 1);
  71. }
  72. void pushup(int rt) {
  73. sum[rt] = sum[rt << 1] + sum[rt << 1 | 1];
  74. }
  75. void pushdown(int rt, int mid) {
  76. if (lazy[rt]) {
  77. lazy[rt << 1] += lazy[rt];
  78. lazy[rt << 1 | 1] += lazy[rt];
  79. sum[rt << 1] += lazy[rt] * (mid - mid / 2);
  80. sum[rt << 1 | 1] += lazy[rt] * (mid / 2);
  81. lazy[rt] = 0;
  82. }
  83. }
  84. void build(int l, int r, int rt) {
  85. lazy[rt] = 0;
  86. if (l == r) {
  87. sum[rt] = a[Rank[l]];
  88. return;
  89. }
  90. int mid = (l + r) >> 1;
  91. build(lson);
  92. build(rson);
  93. pushup(rt);
  94. }
  95. void update(int L, int R, int val, int l, int r, int rt) {
  96. if (L <= l && r <= R) {
  97. sum[rt] += val * (r - l + 1);
  98. lazy[rt] += val;
  99. return;
  100. }
  101. pushdown(rt, r - l + 1);
  102. int mid = (l + r) >> 1;
  103. if (L <= mid) update(L, R, val, lson);
  104. if (mid < R) update(L, R, val, rson);
  105. pushup(rt);
  106. }
  107. int query(int pos, int l, int r, int rt) {
  108. if (l == r) {
  109. return sum[rt];
  110. }
  111. pushdown(rt, r - l + 1);
  112. int mid = (l + r) >> 1;
  113. if (pos <= mid) return query(pos, lson);
  114. if (mid < pos) return query(pos, rson);
  115. }
  116. void change(int x, int y, int val) {
  117. while (top[x] != top[y]) {
  118. if (dep[top[x]] < dep[top[y]]) std::swap(x, y);
  119. update(id[top[x]], id[x], val, 1, n, 1);
  120. x = fa[top[x]];
  121. }
  122. if (dep[x] > dep[y]) std::swap(x, y);
  123. update(id[x], id[y], val, 1, n, 1);
  124. }
  125. int main() {
  126. #ifndef ONLINE_JUDGE
  127. FIN
  128. #endif
  129. while (~scanf("%d %d %d", &n, &m, &Q)) {
  130. init();
  131. build(1, n, 1);
  132. while (Q--) {
  133. char s[2];
  134. int x, y, z;
  135. scanf("%s", s);
  136. if (s[0] == 'I') {
  137. scanf("%d %d %d", &x, &y, &z);
  138. change(x, y, z);
  139. }
  140. if (s[0] == 'D') {
  141. scanf("%d %d %d", &x, &y, &z);
  142. change(x, y, -z);
  143. }
  144. if (s[0] == 'Q') {
  145. scanf("%d", &x);
  146. printf("%d\n", query(id[x], 1, n, 1));
  147. }
  148. }
  149. }
  150. }

POJ2763 Housewife Wind 树链剖分 边权的更多相关文章

  1. POJ - 2763 Housewife Wind (树链剖分/ LCA+RMQ+树状数组)

    题意:有一棵树,每条边给定初始权值.一个人从s点出发.支持两种操作:修改一条边的权值:求从当前位置到点u的最短路径. 分析:就是在边可以修改的情况下求树上最短路.如果不带修改的话,用RMQ预处理LCA ...

  2. POJ 2763 Housewife Wind (树链剖分 有修改单边权)

    题目链接:http://poj.org/problem?id=2763 n个节点的树上知道了每条边权,然后有两种操作:0操作是输出 当前节点到 x节点的最短距离,并移动到 x 节点位置:1操作是第i条 ...

  3. poj 2763 Housewife Wind : 树链剖分维护边 O(nlogn)建树 O((logn)²)修改与查询

    /** problem: http://poj.org/problem?id=2763 **/ #include<stdio.h> #include<stdlib.h> #in ...

  4. BZOJ 1036 [ZJOI2008]树的统计Count (树链剖分 - 点权剖分 - 单点权修改)

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1036 树链剖分模版题,打的时候注意点就行.做这题的时候,真的傻了,单词拼错检查了一个多小时 ...

  5. 计蒜客 38229.Distance on the tree-1.树链剖分(边权)+可持久化线段树(区间小于等于k的数的个数)+离散化+离线处理 or 2.树上第k大(主席树)+二分+离散化+在线查询 (The Preliminary Contest for ICPC China Nanchang National Invitational 南昌邀请赛网络赛)

    Distance on the tree DSM(Data Structure Master) once learned about tree when he was preparing for NO ...

  6. POJ3237 Tree 树链剖分 边权

    POJ3237 Tree 树链剖分 边权 传送门:http://poj.org/problem?id=3237 题意: n个点的,n-1条边 修改单边边权 将a->b的边权取反 查询a-> ...

  7. HDU3669 Aragorn's Story 树链剖分 点权

    HDU3669 Aragorn's Story 树链剖分 点权 传送门:http://acm.hdu.edu.cn/showproblem.php?pid=3966 题意: n个点的,m条边,每个点都 ...

  8. poj 2763 Housewife Wind(树链拆分)

    id=2763" target="_blank" style="">题目链接:poj 2763 Housewife Wind 题目大意:给定一棵 ...

  9. BZOJ 1984: 月下“毛景树” [树链剖分 边权]

    1984: 月下“毛景树” Time Limit: 20 Sec  Memory Limit: 64 MBSubmit: 1728  Solved: 531[Submit][Status][Discu ...

随机推荐

  1. LeetCode21 Merge Two Sorted Lists

    题意: Merge two sorted linked lists and return it as a new list. The new list should be made by splici ...

  2. 2017校赛 问题 D: 我知道了,你知道了吗?【递归】

    题目描述 Alice和Bob走在去学校的路上,听到两个路人的对话: 路人甲:我知道了, 你知道了吗? 路人乙:我知道你知道了,你知道了吗? 路人甲:我知道你知道我知道了,你知道了吗? 路人乙:我知道你 ...

  3. 利用idea构建hibernate

    1.创建项目 若勾选Use library,则点击右侧的Create,使用本地已下载的Hibernate 5.2.13框架(必须导入hibernate-release-5.2.13.Final\lib ...

  4. 巨蟒python全栈开发-第11阶段 ansible3_2入门八个模块

    大纲: 1.file模块 2.fetch模块 3.yum&&pip模块 4.service模块 5.cron模块 6.user模块 7.group模块

  5. 是时候了解React Native了

    文章首发于简书,欢迎关注 随着科技的发展,手机开发也在向好的方向不停的转变.IOS和Android两大手机操作横空出世,称霸江湖.我们每开发一个手机软件最少都需要开发这两个终端. 两大操作系统都在不断 ...

  6. 原生JS使用Blob导出csv文件

    最近在做关于文件下载的需求:前端调用接口,然后对返回数据进行过滤.格式化,然后按表格内容拼接生成csv文件,让用户下载. 具体实现方式如下:let sourceData = { head: [ '时间 ...

  7. golang gin框架 使用swagger生成api文档

    github地址:https://github.com/swaggo/gin-swagger 1.下载swag $ go get -u github.com/swaggo/swag/cmd/swag ...

  8. 前端知识---html

    HTML HTML是英文Hyper Text Mark-up Language(超文本标记语言)的缩写,他是一种制作万维网页面标准语言(标记).相当于定义统一的一套规则,大家都来遵守他,这样就可以让浏 ...

  9. 使用HSV色彩空间遮罩绿色区域

    HSV 颜色空间 导入资源 In []: import matplotlib.pyplot as plt import matplotlib.image as mpimg ​ import numpy ...

  10. 3、.net core 部署到IIS

    1)下载对应的Hosting Bundle https://dotnet.microsoft.com/download/dotnet-core/2.2 2)VS发布项目,选择window平台环境 3 ...