题目大意

给出一个序列和\(L, R\), 求前k大长度在\([L,R]\)之间的连续子序列的和的和.

解题思路

朴素的想法是对于一个左端点\(p\), 它的右区间取值范围是一个连续的区间即\([p+L-1,p+R-1]\). 枚举这些区间的和然后排序一下什么的, 当然可以用前缀和优化.

考虑对于一个左端点和一个右端点的区间组成的三元组, 设它的答案是\(f(p,l,r)\),那么\(f(p,l,r)\)是一个固定的数, 即\(f(p,l,r)=max(sum(p,r)), \text{其中}(r\in [p+L-1,p+R-1])\).

转化为前缀和后, \(sum(p,r)=sum_r-sum_{p-1}\), 由于\(sum_{p-1}\)一定, 其实就是\(sum_r\)最大的\(r\)最优. 于是这样的最优的\(r\)可以用ST表在\(O(1)\)的时间内求出.

实际上对于每一个左端点都有一个类似的三元组, 考虑如何能把最大的前\(k\)个子序列求出.

若此时\(f(p,l,r)\)(设\([l,r]\)中最优的端点是\(w\))最大, 将\(f(p,l,r)\)累加到答案的同时, \([l,r]\)这个右端点区间会分裂成两个区间\([l,w)\)和\((w,r]\)(前提是这样的区间存在).

为什么? \(w\)只是\([l,r]\)中最大的右端点, 实际上次大的, 第三大的..都能作为可能的答案.

根据上面的分析, 解法已经呼然欲出: 我们维护一个大根堆, 每次取出最大的三元组, 累加答案并分裂区间加入堆. 由于一共只会分裂\(k\)次的区间, 所以堆中的元素不超过\(n+k\)个.

加上预处理, 时间复杂度\(O(n\log{n}+k\log{(n+k)})\).

后记

自己的数据结构真的好差.. 某些大佬(pzr等人)用主席树也能过, 我又不会....

PS:洛谷上的数据范围比标程要大一些

  1. #include <queue>
  2. #include <cstdio>
  3. #include <cstring>
  4. #define W 17
  5. #define N 100010
  6. #define ll long long
  7. #define fo(i, a, b) for(int i = (a); i <= (b); ++i)
  8. #define fd(i, a, b) for(int i = (a); i >= (b); --i)
  9. using namespace std;
  10. inline int min(int a, int b){return a < b ? a : b;}
  11. inline int read()
  12. {
  13. int x = 0; char ch = getchar(); bool ne = 0;
  14. while(ch < '0' || ch > '9') ne |= (ch == '-'), ch = getchar();
  15. while(ch >= '0' && ch <= '9') x = (x << 3) + (x << 1) + (ch ^ 48), ch = getchar();
  16. return ne ? -x : x;
  17. }
  18. int n, k, L, R, lg[N], p2[W + 3] = {1}, sum[N];
  19. ll ans;
  20. namespace ST
  21. {
  22. int s[N][W + 3];
  23. void init()
  24. {
  25. fo(i, 1, n) s[i][0] = i;
  26. fo(j, 1, lg[n])
  27. fo(i, 1, n - p2[j] + 1)
  28. {
  29. int x = s[i][j - 1], y = s[i + p2[j - 1]][j - 1];
  30. s[i][j] = sum[x] > sum[y] ? x : y;
  31. }
  32. }
  33. inline int query(int l, int r)
  34. {
  35. int p = lg[r - l + 1];
  36. int x = s[l][p], y = s[r - p2[p] + 1][p];
  37. return sum[x] > sum[y] ? x : y;
  38. }
  39. }
  40. struct Node
  41. {
  42. int p, l, r, w;
  43. Node(){}
  44. Node(int _p, int _l, int _r){p = _p, l = _l, r = _r, w = ST::query(l, r);}
  45. inline int val() const {return sum[w] - sum[p - 1];}
  46. bool operator<(const Node b) const
  47. {
  48. return val() < b.val();
  49. }
  50. };
  51. priority_queue<Node> h;
  52. int main()
  53. {
  54. freopen("fantasy.in", "r", stdin);
  55. freopen("fantasy.out", "w", stdout);
  56. n = read(), k = read(), L = read(), R = read();
  57. fo(i, 1, W) p2[i] = p2[i - 1] << 1;
  58. fo(i, 1, n) sum[i] = read() + sum[i - 1];
  59. fo(i, 2, n) lg[i] = lg[i >> 1] + 1;
  60. ST::init();
  61. fo(i, 1, n - L + 1)
  62. h.push(Node(i, i + L - 1, min(n, i + R - 1)));
  63. fo(i, 1, k)
  64. {
  65. Node t = h.top(); h.pop();
  66. ans += t.val();
  67. if(t.l < t.w) h.push(Node(t.p, t.l, t.w - 1));
  68. if(t.w < t.r) h.push(Node(t.p, t.w + 1, t.r));
  69. }
  70. printf("%lld", ans);
  71. return 0;
  72. }

JZOJ5409. Fantasy && Luogu2048 [NOI2010]超级钢琴的更多相关文章

  1. luogu2048 [NOI2010]超级钢琴 (优先队列+主席树)

    思路:先扫一遍所有点作为右端点的情况,把它们能产生的最大值加到一个优先队列里,然后每次从优先队列里取出最大值,再把它对应的区间的次大值加到优先队列里,这样做K次 可以用一个前缀和,每次找i为右端点的第 ...

  2. BZOJ 2006: [NOI2010]超级钢琴

    2006: [NOI2010]超级钢琴 Time Limit: 20 Sec  Memory Limit: 552 MBSubmit: 2613  Solved: 1297[Submit][Statu ...

  3. Bzoj 2006: [NOI2010]超级钢琴 堆,ST表

    2006: [NOI2010]超级钢琴 Time Limit: 20 Sec  Memory Limit: 552 MBSubmit: 2222  Solved: 1082[Submit][Statu ...

  4. NOI2010超级钢琴 2

    2006: [NOI2010]超级钢琴 Time Limit: 20 Sec  Memory Limit: 552 MBSubmit: 1296  Solved: 606[Submit][Status ...

  5. BZOJ 2006: [NOI2010]超级钢琴( RMQ + 堆 )

    取最大的K个, 用堆和RMQ来加速... ----------------------------------------------------------------- #include<c ...

  6. BZOJ_2006_[NOI2010]超级钢琴_贪心+堆+ST表

    BZOJ_2006_[NOI2010]超级钢琴_贪心+堆+ST表 Description 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的 音乐 ...

  7. bzoj2006 [NOI2010]超级钢琴 (及其拓展)

    bzoj2006 [NOI2010]超级钢琴 给定一个序列,求长度在 \([L,\ R]\) 之间的区间和的前 \(k\) 大之和 \(n\leq5\times10^5,\ k\leq2\times1 ...

  8. P2048 [NOI2010]超级钢琴(RMQ+堆+贪心)

    P2048 [NOI2010]超级钢琴 区间和--->前缀和做差 多次查询区间和最大--->前缀和RMQ 每次取出最大的区间和--->堆 于是我们设个3元组$(o,l,r)$,表示左 ...

  9. 洛谷 P2048 [NOI2010]超级钢琴 解题报告

    P2048 [NOI2010]超级钢琴 题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为 ...

随机推荐

  1. Advanced C++ | Virtual Copy Constructor

    这个不懂,等看会了再写...

  2. java foreach循环抛出异常java.util.ConcurrentModificationException

    代码如下: for (Iterator<String> iter = list.iterator(); iter.hasNext(); ) { if (Integer.parseInt(i ...

  3. Spring Cloud中,如何解决Feign整合Hystrix第一次请求失败的问题

    Spring Cloud中,Feign和Ribbon在整合了Hystrix后,可能会出现首次调用失败的问题,要如何解决该问题呢? 造成该问题的原因 Hystrix默认的超时时间是1秒,如果超过这个时间 ...

  4. 【Xcode】sh: pause: command not found

    system("pause"); 只适合于DOS和Windows系统,不适合Linux系统. 直接删掉就可以. 或者改为: #include <unistd.h> pa ...

  5. 什么是mysql innodb cluster?

    目录 一.简介 二.特性 一.简介 MySQL InnoDB集群提供了一个集成的,本地的,HA解决方案.MySQL InnoDB集群由以下几部分组成: MySQL Servers with Group ...

  6. jenkins pipeline语法

    目录 一.声明式 二.脚本式 基本 判断 异常处理 Steps node withEnv 一.声明式 声明式Pipeline必须包含在名为pipeline的语句块中,典型的声明式Pipeline语法如 ...

  7. 攻击科普:ARP攻击

    目录 一.介绍 二.解决办法 一.介绍 ARP攻击的局限性 ARP攻击仅能在以太网(局域网如:机房.内网.公司网络等)进行. 无法对外网(互联网.非本区域内的局域网)进行攻击. ARP攻击就是通过伪造 ...

  8. [BUUCTF]REVERSE——[2019红帽杯]easyRE

    [2019红帽杯]easyRE 附件 步骤: ida载入,没有main函数,就先检索了程序里的字符串 发现了base64加密的特征字符串,双击you found me跟进,找到了调用它的函数,函数很长 ...

  9. [BUUCTF]PWN——jarvisoj_level3

    jarvisoj_level3 附件 步骤 例行检查,32位,nx保护 运行一下程序 32位ida载入,shift+f12没有看到程序里有可以直接利用的后面函数,根据运行时的字符串找到了程序的关键函数 ...

  10. xmake v2.6.2 发布,新增 Linux 内核驱动模块构建支持

    Xmake 是一个基于 Lua 的轻量级跨平台构建工具. 它非常的轻量,没有任何依赖,因为它内置了 Lua 运行时. 它使用 xmake.lua 维护项目构建,相比 makefile/CMakeLis ...