题目传送门

https://lydsy.com/JudgeOnline/problem.php?id=4009

题解

考虑怎样的情况就会有一个链覆盖另一个链。

设被覆盖的链为 \(a - b\),覆盖的链为 \(x - y\)。假设有 \(dfn[a] < dfn[b], dfn[x] < dfn[y]\)

那么如果 \(a\) 是 \(b\) 的祖先,那么令 \(g\) 为 \(a\) 的子树中包含 \(b\) 的点,那么 \(x, y\) 中有一个点在 \(g\) 的子树外面,一个在 \(b\) 子树里面,即 \(x \in [1, dfn[g] - 1], y \in [dfn[b], dfn[b] + siz[b] - 1]\),\(x \in [dfn[b], dfn[b] + siz[b] - 1], y \in [dfn[g] + siz[g], n]\)。

否则,那么 \(x, y\) 必然一个在 \(a\) 子树中,一个在 \(b\) 子树中。即 \(x \in [dfn[a], dfn[a] + siz[a] - 1]\),\(y \in [dfn[b], dfn[b] + siz[b] - 1]\)。

那么我们不妨把上面的问题看成求出包含一个点的矩阵的权值的第 \(k\) 小。

这个问题似乎可以用扫描线+动态主席树来做。(就是带修改区间 \(k\) 小值啊)

但是应该整体二分更好写一写,整体二分以后转化为求出一个点被几个矩形覆盖,也是扫描线+树状数组维护。


但是我似乎已经忘了整体二分怎么写了,然后学了很久,写的时候写出了一堆的 bug,调了一整天才调完。

不过这个题目把询问转化为矩形维护的相关操作是一个很好的思想。


时间复杂度 \(O(q\log^2n)\)。

  1. #include<bits/stdc++.h>
  2. #define fec(i, x, y) (int i = head[x], y = g[i].to; i; i = g[i].ne, y = g[i].to)
  3. #define dbg(...) fprintf(stderr, __VA_ARGS__)
  4. #define File(x) freopen(#x".in", "r", stdin), freopen(#x".out", "w", stdout)
  5. #define fi first
  6. #define se second
  7. #define pb push_back
  8. template<typename A, typename B> inline char smax(A &a, const B &b) {return a < b ? a = b, 1 : 0;}
  9. template<typename A, typename B> inline char smin(A &a, const B &b) {return b < a ? a = b, 1 : 0;}
  10. typedef long long ll; typedef unsigned long long ull; typedef std::pair<int, int> pii;
  11. template<typename I> inline void read(I &x) {
  12. int f = 0, c;
  13. while (!isdigit(c = getchar())) c == '-' ? f = 1 : 0;
  14. x = c & 15;
  15. while (isdigit(c = getchar())) x = (x << 1) + (x << 3) + (c & 15);
  16. f ? x = -x : 0;
  17. }
  18. const int N = 40000 + 7;
  19. #define lowbit(x) ((x) & -(x))
  20. int n, m, Q, dfc, cnt;
  21. int dep[N], f[N], siz[N], son[N], top[N], dfn[N], pre[N], ans[N];
  22. std::priority_queue<std::pair<int, pii> > q;
  23. struct Matrix {
  24. int x1, x2, y1, y2, v;
  25. inline bool operator < (const Matrix &b) const { return x1 < b.x1; }
  26. } a[N << 1], a1[N << 1], a2[N << 1];
  27. struct Point {
  28. int x, y, k, *ans;
  29. inline bool operator < (const Point &b) const { return x < b.x; }
  30. } b[N], b1[N], b2[N];
  31. struct Edge { int to, ne; } g[N << 1]; int head[N], tot;
  32. inline void addedge(int x, int y) { g[++tot].to = y, g[tot].ne = head[x], head[x] = tot; }
  33. inline void adde(int x, int y) { addedge(x, y), addedge(y, x); }
  34. inline void dfs1(int x, int fa = 0) {
  35. dep[x] = dep[fa] + 1, f[x] = fa, siz[x] = 1;
  36. for fec(i, x, y) if (y != fa) dfs1(y, x), siz[x] += siz[y], siz[y] > siz[son[x]] && (son[x] = y);
  37. }
  38. inline void dfs2(int x, int pa) {
  39. top[x] = pa, dfn[x] = ++dfc, pre[dfc] = x;
  40. if (!son[x]) return; dfs2(son[x], pa);
  41. for fec(i, x, y) if (y != f[x] && y != son[x]) dfs2(y, y);
  42. }
  43. inline int gson(int x, int p) {
  44. int g = 0;
  45. while (top[x] != top[p]) g = top[x], x = f[g];
  46. return x == p ? g : son[p];
  47. }
  48. inline bool intr(int x, int p) { return dfn[x] >= dfn[p] && dfn[x] <= dfn[p] + siz[p] - 1; }
  49. namespace BIT {
  50. int s[N];
  51. inline void qadd(int x, int k) { for (; x <= n; x += lowbit(x)) s[x] += k; }
  52. inline int qsum(int x) {
  53. int ans = 0;
  54. for (; x; x -= lowbit(x)) ans += s[x];
  55. return ans;
  56. }
  57. inline void qadd(int l, int r, int k) { qadd(l, k), qadd(r + 1, -k); }
  58. }
  59. using BIT::qadd;
  60. using BIT::qsum;
  61. inline void solve(int L, int R, int st, int ed, int l, int r) {
  62. if (l > r) return;
  63. if (L == R) {
  64. for (int i = l; i <= r; ++i) *b[i].ans = L;
  65. return;
  66. }
  67. int M = (L + R) >> 1, n1 = 0, n2 = 0, j = l - 1;
  68. for (int i = st; i <= ed; ++i) {
  69. if (a[i].v <= M) {
  70. qadd(a[i].y1, a[i].y2, 1);
  71. q.push(std::make_pair(-a[i].x2, pii(a[i].y1, a[i].y2)));
  72. }
  73. if (i != ed && a[i].x1 == a[i + 1].x1) continue;
  74. while (j < r && (i == ed || b[j + 1].x < a[i + 1].x1)) {
  75. ++j;
  76. while (!q.empty() && -q.top().fi < b[j].x) qadd(q.top().se.fi, q.top().se.se, -1), q.pop();
  77. int cnt = qsum(b[j].y);
  78. if (cnt >= b[j].k) b1[++n1] = b[j];
  79. else b2[++n2] = b[j], b2[n2].k -= cnt;
  80. }
  81. }
  82. while (!q.empty()) qadd(q.top().se.fi, q.top().se.se, -1), q.pop();
  83. int m1 = 0, m2 = 0;
  84. for (int i = st; i <= ed; ++i) if (a[i].v <= M) a1[++m1] = a[i]; else a2[++m2] = a[i];
  85. assert(m1 + m2 == ed - st + 1), assert(n1 + n2 == r - l + 1);
  86. std::copy(a1 + 1, a1 + m1 + 1, a + st), std::copy(a2 + 1, a2 + m2 + 1, a + st + m1);
  87. std::copy(b1 + 1, b1 + n1 + 1, b + l), std::copy(b2 + 1, b2 + n2 + 1, b + l + n1);
  88. solve(L, M, st, st + m1 - 1, l, l + n1 - 1), solve(M + 1, R, st + m1, ed, l + n1, r);
  89. }
  90. inline void work() {
  91. std::sort(a + 1, a + cnt + 1);
  92. std::sort(b + 1, b + Q + 1);
  93. solve(0, 1e9, 1, cnt, 1, Q);
  94. for (int i = 1; i <= Q; ++i) printf("%d\n", ans[i]);
  95. }
  96. inline void init() {
  97. read(n), read(m), read(Q);
  98. int x, y;
  99. for (int i = 1; i < n; ++i) read(x), read(y), adde(x, y);
  100. dfs1(1), dfs2(1, 1);
  101. for (int i = 1; i <= m; ++i) {
  102. int x, y, k;
  103. read(x), read(y), read(k);
  104. assert(x != y);
  105. if (dfn[x] > dfn[y]) std::swap(x, y);
  106. if (intr(y, x)) {
  107. x = gson(y, x);
  108. if (dfn[x] > 1) a[++cnt] = (Matrix){ 1, dfn[x] - 1, dfn[y], dfn[y] + siz[y] - 1, k };
  109. if (dfn[x] + siz[x] - 1 < n) a[++cnt] = (Matrix){ dfn[y], dfn[y] + siz[y] - 1, dfn[x] + siz[x], n, k };
  110. } else a[++cnt] = (Matrix){ dfn[x], dfn[x] + siz[x] - 1, dfn[y], dfn[y] + siz[y] - 1, k };
  111. }
  112. for (int i = 1; i <= Q; ++i) {
  113. int x, y, k;
  114. read(x), read(y), read(k);
  115. if (dfn[x] > dfn[y]) std::swap(x, y);
  116. b[i] = (Point){ dfn[x], dfn[y], k, ans + i };
  117. }
  118. }
  119. int main() {
  120. #ifdef hzhkk
  121. freopen("hkk.in", "r", stdin);
  122. #endif
  123. init();
  124. work();
  125. fclose(stdin), fclose(stdout);
  126. return 0;
  127. }

bzoj4009 [HNOI2015]接水果 整体二分+扫描线+树状数组+dfs序的更多相关文章

  1. BZOJ 4009: [HNOI2015]接水果 (整体二分+扫描线 树状数组)

    整体二分+扫描线 树状数组 具体做法看这里a CODE #include <cctype> #include <cstdio> #include <cstring> ...

  2. 【BZOJ4009】[HNOI2015]接水果 DFS序+整体二分+扫描线+树状数组

    [BZOJ4009][HNOI2015]接水果 Description 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果.由于她已经DT FC 了The big black, ...

  3. [bzoj4009] [HNOI2015]接水果 整体二分+扫描线+dfs序+树状数组

    Description 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果. 由于她已经DT FC 了The big black, 她觉得这个游戏太简单了,于是发明了一个更 加 ...

  4. 【BZOJ】2434: [Noi2011]阿狸的打字机 AC自动机+树状数组+DFS序

    [题意]阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的: l 输入小写 ...

  5. [BZOJ4009][HNOI2015]接水果(整体二分)

    [HNOI2015]接水果 时间限制:60s      空间限制:512MB 题目描述 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果. 由于她已经DT FC 了The b ...

  6. BZOJ.4009.[HNOI2015]接水果(整体二分 扫描线)

    LOJ BZOJ 洛谷 又是一个三OJ rank1!=w= \(Description\) (还是感觉,为啥非要出那种题目背景啊=-=直接说不好么) 给定一棵树和一个路径集合(每条路径有一个权值).\ ...

  7. Luogu3527 POI2011 Meteors 整体二分、树状数组、差分

    传送门 比较板子的整体二分题目,时限有点紧注意常数 整体二分的过程中将时间在\([l,mid]\)之间的流星使用树状数组+差分进行维护,然后对所有国家查看一遍并分好类,递归下去,记得消除答案在\([m ...

  8. [bzoj2527][Poi2011]Meteors_整体二分_树状数组

    Meteors bzoj-2527 Poi-2011 题目大意:题目链接. 注释:略. 想法: 首先答案可以离线,且具有单调性. 这里的单调性就是随着时间的推移,每个国家收集的陨石数增加. 不难想到整 ...

  9. [bzoj2738]矩阵乘法_整体二分_树状数组

    矩阵乘法 bzoj-2738 题目大意:给定一个$n*n$的矩阵.每次给定一个矩阵求矩阵$k$小值. 注释:$1\le n\le 500$,$1\le q\le 6\cdot 10^4$. 想法: 新 ...

随机推荐

  1. 《SQL Server 2012 T-SQL基础》读书笔记 - 4.子查询

    Chapter 4 Subqueries 子查询分为:独立子查询(Self-Contained Subqueries)和相关子查询(Correlated Subqueries),独立子查询可以单独拿出 ...

  2. axios的详细用法以及后端接口代理

    安装 使用 npm: $ npm install axios 或者 使用 bower: $ bower install axios 或者直接使用 cdn: <script src="h ...

  3. centos输入正确的账号和密码登陆不进去

    vm下启动centos,输入正确的账号和密码,依然登陆不进去,一直处于这个界面: 暂时的解决方法是:先等待一段时间.重启,然后再输入密码,然后,ctrl+c 不停地ctrl+c,然后就登陆进去了.什么 ...

  4. day51—JavaScript绑定事件

    转换学开发,代码100天——2018-05-06 今天学习JavaScript的绑定事件.因为浏览器的原因绑定事件需要考虑兼容性问题. attachEvent     IE浏览器 ,ie9以上事件执行 ...

  5. day50—JavaScript鼠标拖拽事件

    转行学开发,代码100天——2018-05-05 今天通过鼠标拖拽事件复习巩固一下鼠标事件. 鼠标拖拽事件需要记住两点: 1.距离不变 2.鼠标事件(按下,移动,抬起) <div id=&quo ...

  6. 使用@Value注解对bean进行属性注入

    使用@Value注解,可以有三种属性注入的方式: 1. 使用字面量注入 2. 使用EL表达式注入 3. 使用占位符注入 import org.springframework.beans.factory ...

  7. android dialog,popupwindow,toast窗口的添加机制

    Dialog 窗口添加机制 代码示例 首先举两个例子: 例子1 在Activity中 @OnClick(R.id.but) void onClick() { Log.d("LiaBin&qu ...

  8. python接口自动化:调试接口的代码(无token情况下)

    实现代码如下: #接口调试 import requests,time class api_demo1: def __init__(self,RequestWay,url,data): self.s=r ...

  9. 12 (H5*) JS第二天 流程控制:顺序结构、分支结构、循环结构

    目录 1:一元运算符 2:流程控制 3:分支之if语句 4:分支之if-else语句 5:分支语句之三元运算符 6:if和else if语句 7:switch-case语句 8:while循环 9:d ...

  10. hackinglab 基础关 writeup

    地址:http://hackinglab.cn/ 基础关 key在哪里? 很简单,点击过关地址,在新打开的网页中查看网页源代码就能在 HTML 注释中发现 key 再加密一次你就得到key啦~ 明文加 ...