• CDQ分治+树状数组,LOJ6270. 数据结构板子题


    一、题目

    1、题目描述

    有 n 个区间,第 i 个区间是 [l_i,r_i],它的长度是 r_i-l_i。

    有 q 个询问,每个询问给定 L,R,K,询问被 [L,R] 包含的且长度不小于 K 的区间数量。

    你想,像这种板子题,你随手写,不到十分钟就能 AC。

    2、输入输出

    2.1输入

    第一行,两个空格隔开的正整数 n,q。

    接下来 n 行,第 i 行有两个空格隔开的正整数 l_i,r_i。

    接下来 q 行,每行三个空格隔开的正整数 L,R,K,表示一个询问。

    2.2输出

    共 q 行,每行一个非负整数,表示询问的答案。

    3、原题链接

    #6270. 数据结构板子题 - 题目 - LibreOJ (loj.ac)


    二、解题报告

    1、思路分析

    看到这种区间问题,不难想到应该要用分治

    我们考虑[l, r]内长度不小于k的区间数目 = [l, r]内总区间数目 - [l, r]内小于k的区间数目

    那么我们先读入区间

    然后读入询问,我们把每个询问[l, r, k]拆成两个,一个是[l, r, k - 1],一个是[l, r, r - l]

    后者减去前者就是答案

    具体操作时,我们把区间和询问都按照区间长度升序排序,然后开两个树状数组,一个维护左端点左边的区间数目,一个维护右端点左边的区间数目

    然后双指针归并,计算贡献即可

    2、复杂度

    时间复杂度:O(nlog^2 n) 空间复杂度:O(n)

    3、代码详解

    1. #include
    2. #include
    3. #include
    4. #include
    5. using namespace std;
    6. const int N = 5e5 + 10, inf = 1e9 + 7;
    7. struct line
    8. {
    9. int l, r, len;
    10. bool operator<(const line &x) const
    11. {
    12. return len < x.len;
    13. }
    14. } lines[N];
    15. struct query
    16. {
    17. int l, r, len, id;
    18. bool operator<(const query &x) const
    19. {
    20. return len < x.len;
    21. }
    22. } query[N << 1];
    23. int tot = 0, n, m, ans[N << 1], trR[N], trL[N];
    24. void addR(int x, int v)
    25. {
    26. for (; x <= n; x += (x & -x))
    27. trR[x] += v;
    28. }
    29. void addL(int x, int v)
    30. {
    31. for (; x <= n; x += (x & -x))
    32. trL[x] += v;
    33. }
    34. int askR(int x)
    35. {
    36. int res = 0;
    37. for (; x > 0; x &= (x - 1))
    38. res += trR[x];
    39. return res;
    40. }
    41. int askL(int x)
    42. {
    43. int res = 0;
    44. for (; x > 0; x &= (x - 1))
    45. res += trL[x];
    46. return res;
    47. }
    48. int main()
    49. {
    50. //freopen("in.txt", "r", stdin);
    51. ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
    52. cin >> n >> m;
    53. for (int i = 1, l, r; i <= n; i++)
    54. cin >> l >> r, lines[i] = {l, r, r - l};
    55. for (int i = 1, l, r, k; i <= m; i++)
    56. {
    57. cin >> l >> r >> k;
    58. ++tot, query[tot] = {l, r, k - 1, tot};
    59. ++tot, query[tot] = {l, r, r - l, tot};
    60. }
    61. sort(lines + 1, lines + 1 + n), sort(query + 1, query + 1 + tot);
    62. for (int i = 1, p = 1; i <= tot; i++)
    63. {
    64. while (p <= n && lines[p].len <= query[i].len)
    65. addL(lines[p].l, 1), addR(lines[p].r, 1), p++;
    66. if (query[i].len <= query[i].r - query[i].l)
    67. ans[query[i].id] = askR(query[i].r) - askL(query[i].l - 1);
    68. else
    69. ans[query[i].id] = inf;
    70. }
    71. for (int i = 1; i <= m; i++)
    72. cout << max(0, ans[i << 1] - ans[(i << 1) - 1]) << '\n';
    73. return 0;
    74. }

  • 相关阅读:
    知识图谱1(实体抽取)
    基于软约束的轨迹(路径)优化原理公式推导详解
    一些封装好、使用度高的Api(JavaScript)
    鲸鱼算法优化LSTM超参数-神经元个数-dropout-batch_size
    Linux网络——HTTP
    教你自己制作一个ALU
    Vue2 Element Pagination组件 每页数据量不同的解决方案
    Android开发之——Jetpack Compose布局(03)
    MYSQL指令
    Stable Diffusion原理
  • 原文地址:https://blog.csdn.net/EQUINOX1/article/details/136157474