JZOJ5409. Fantasy && Luogu2048 [NOI2010]超级钢琴
题目大意
给出一个序列和\(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:洛谷上的数据范围比标程要大一些
#include <queue>
#include <cstdio>
#include <cstring>
#define W 17
#define N 100010
#define ll long long
#define fo(i, a, b) for(int i = (a); i <= (b); ++i)
#define fd(i, a, b) for(int i = (a); i >= (b); --i)
using namespace std;
inline int min(int a, int b){return a < b ? a : b;}
inline int read()
{
int x = 0; char ch = getchar(); bool ne = 0;
while(ch < '0' || ch > '9') ne |= (ch == '-'), ch = getchar();
while(ch >= '0' && ch <= '9') x = (x << 3) + (x << 1) + (ch ^ 48), ch = getchar();
return ne ? -x : x;
}
int n, k, L, R, lg[N], p2[W + 3] = {1}, sum[N];
ll ans;
namespace ST
{
int s[N][W + 3];
void init()
{
fo(i, 1, n) s[i][0] = i;
fo(j, 1, lg[n])
fo(i, 1, n - p2[j] + 1)
{
int x = s[i][j - 1], y = s[i + p2[j - 1]][j - 1];
s[i][j] = sum[x] > sum[y] ? x : y;
}
}
inline int query(int l, int r)
{
int p = lg[r - l + 1];
int x = s[l][p], y = s[r - p2[p] + 1][p];
return sum[x] > sum[y] ? x : y;
}
}
struct Node
{
int p, l, r, w;
Node(){}
Node(int _p, int _l, int _r){p = _p, l = _l, r = _r, w = ST::query(l, r);}
inline int val() const {return sum[w] - sum[p - 1];}
bool operator<(const Node b) const
{
return val() < b.val();
}
};
priority_queue<Node> h;
int main()
{
freopen("fantasy.in", "r", stdin);
freopen("fantasy.out", "w", stdout);
n = read(), k = read(), L = read(), R = read();
fo(i, 1, W) p2[i] = p2[i - 1] << 1;
fo(i, 1, n) sum[i] = read() + sum[i - 1];
fo(i, 2, n) lg[i] = lg[i >> 1] + 1;
ST::init();
fo(i, 1, n - L + 1)
h.push(Node(i, i + L - 1, min(n, i + R - 1)));
fo(i, 1, k)
{
Node t = h.top(); h.pop();
ans += t.val();
if(t.l < t.w) h.push(Node(t.p, t.l, t.w - 1));
if(t.w < t.r) h.push(Node(t.p, t.w + 1, t.r));
}
printf("%lld", ans);
return 0;
}
JZOJ5409. Fantasy && Luogu2048 [NOI2010]超级钢琴的更多相关文章
- luogu2048 [NOI2010]超级钢琴 (优先队列+主席树)
思路:先扫一遍所有点作为右端点的情况,把它们能产生的最大值加到一个优先队列里,然后每次从优先队列里取出最大值,再把它对应的区间的次大值加到优先队列里,这样做K次 可以用一个前缀和,每次找i为右端点的第 ...
- BZOJ 2006: [NOI2010]超级钢琴
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MBSubmit: 2613 Solved: 1297[Submit][Statu ...
- Bzoj 2006: [NOI2010]超级钢琴 堆,ST表
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MBSubmit: 2222 Solved: 1082[Submit][Statu ...
- NOI2010超级钢琴 2
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MBSubmit: 1296 Solved: 606[Submit][Status ...
- BZOJ 2006: [NOI2010]超级钢琴( RMQ + 堆 )
取最大的K个, 用堆和RMQ来加速... ----------------------------------------------------------------- #include<c ...
- BZOJ_2006_[NOI2010]超级钢琴_贪心+堆+ST表
BZOJ_2006_[NOI2010]超级钢琴_贪心+堆+ST表 Description 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的 音乐 ...
- bzoj2006 [NOI2010]超级钢琴 (及其拓展)
bzoj2006 [NOI2010]超级钢琴 给定一个序列,求长度在 \([L,\ R]\) 之间的区间和的前 \(k\) 大之和 \(n\leq5\times10^5,\ k\leq2\times1 ...
- P2048 [NOI2010]超级钢琴(RMQ+堆+贪心)
P2048 [NOI2010]超级钢琴 区间和--->前缀和做差 多次查询区间和最大--->前缀和RMQ 每次取出最大的区间和--->堆 于是我们设个3元组$(o,l,r)$,表示左 ...
- 洛谷 P2048 [NOI2010]超级钢琴 解题报告
P2048 [NOI2010]超级钢琴 题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为 ...
随机推荐
- 深入分析 Java ZGC
传统的垃圾回收 CMS 与 G1 停顿时间瓶颈 ZGC 概览 深入 ZGC 原理 标记 Marking 着色指针 Reference Coloring Relocation 重映射和读屏障 Remap ...
- python3.6.4 scrapy框架from PIL import Image报错 from . import _imaging as core
scrapy框架爬取url下载图片时,用ImagesPipeline下载图片 from PIL import Image报错 from . import _imaging as core Import ...
- Mysql配置 主从同步
目录 一.准备 二.操作 主数据库操作 从服务器操作 一.准备 1.主从数据库版本最好一致 2.主从数据库内数据保持一致,若不一致,可将从库中所有数据删除,并将主库全部数据导入进去 主数据库:182. ...
- java 多线程 线程池:多核CPU利用ExecutorService newWorkStealingPool; ForkJoinPool线程池 执行可拆分的任务RecursiveAction;RecursiveTask
1,给定并行级别: 1,ExecutorService newWorkStealingPool(int parallelism): 创建持有足够的线程的线程池来支持给定的并行级别,该方法还会使用多个队 ...
- 升级shiro1.6版本后导致附件上传失败,浏览器返回400错误
最新shiro发布了一个漏洞,凡是jar包在1.6版本的都会出现该漏洞,要修复该漏洞只能升级到shiro1.6版本 但是如果项目中url使用了;jsessionid这种方式的话 就会导致上传失败,浏览 ...
- JAVA整合阿里云OSS实现文件上传功能
引入maven <dependency> <groupId>org.apache.commons</groupId> <artifactId>commo ...
- HDZ城市行深圳站|AIoT时代,如何抓住智联生活的战略机会点?
摘要:2021年12月24日,HDZ城市行深圳站:AIoT引爆全场景应用新机会(智联生活专场)圆满落幕. 2021年12月24日,HDZ城市行深圳站:AIoT引爆全场景应用新机会(智联生活专场)圆满落 ...
- 给初学者的STM32(Cortex-M3)中断原理及编程方法介绍 [原创www.cnblogs.com/helesheng]
本人编著的<基于STM32的嵌入式系统原理及应用>(ISBN:9787030697974)刚刚在科学出版社出版.这本书花费了半年以上的时间,凝聚了笔者作为高校教师和嵌入式工程师的一些经验, ...
- BitBake使用攻略--BitBake的语法知识一
目录 写在前面 1. BitBake中的赋值 1.1 直接赋值 1.2 间接赋值 1.3 追加与前加赋值 1.4 Override风格的赋值语法 1.5 标志赋值 1.6 内联函数赋值 1.7 其他一 ...
- 【LeetCode】235. Lowest Common Ancestor of a Binary Search Tree 解题报告(Java & Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 [LeetCode] https://leet ...