【bzoj2006】[NOI2010]超级钢琴 倍增RMQ+STL-堆
题目描述
输入
输出
只有一个整数,表示乐曲美妙度的最大值。
样例输入
4 3 2 3
3
2
-6
8
样例输出
11
题解
倍增RMQ(貌似这个叫做ST表)+STL-堆
首先我们将连续子序列和转化为前缀和相减,那么对于每个左端点,找它的右端点,就转变为求某范围内前缀和的最大值、次大值、...
但是直接拿出整个序列显然会TLE+MLE。
我们考虑,如果左端点t,右端点[l,r],我们找出属于[l,r]的位置p,使得sum[p]最大。那么很显然,当且仅当p被选择之后,[l,p-1]和[p+1,r]内的右端点才可能会被选择。
所以我们可以维护一个结构体,储存t、l、r、p、v,分别表示左端点、右端点区间的左位置、右端点区间的右位置、右端点区间最大值的位置、右端点区间最大值,显然v=sum[p]-sum[t-1]。我们优先选择v较大的区间把它拿出来,把它的v加入到答案中。然后考虑剩下的[l,p-1]和[p+1,r],如果存在这些区间的话,采取相同的方式处理即可。这个过程显然可以使用堆来维护。
维护区间最大值即位置。可以使用倍增RMQ来解决,时间复杂度更低一些。
因此总的时间复杂度为$O((n+k)\log n)$
另外需要注意的是priority_queue需要重载<运算符,而不是>(尽管它是大根堆)
#include <cstdio>
#include <queue>
#define N 500010
using namespace std;
struct data
{
int t , b , e , v , p;
data() {}
data(int t0 , int b0 , int e0 , int v0 , int p0) {t = t0 , b = b0 , e = e0 , v = v0 , p = p0;}
bool operator<(const data a)const {return v < a.v;}
}tmp;
priority_queue<data> q;
int sum[N] , maxn[N][20] , maxp[N][20] , log[N];
int getmaxn(int x , int y)
{
int k = log[y - x + 1];
return max(maxn[x][k] , maxn[y - (1 << k) + 1][k]);
}
int getmaxp(int x , int y)
{
int k = log[y - x + 1];
return maxn[x][k] > maxn[y - (1 << k) + 1][k] ? maxp[x][k] : maxp[y - (1 << k) + 1][k];
}
int main()
{
int n , k , l , r , i , j , x , y;
long long ans = 0;
scanf("%d%d%d%d" , &n , &k , &l , &r);
for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &x) , sum[i] = sum[i - 1] + x , maxn[i][0] = sum[i] , maxp[i][0] = i;
for(i = 2 ; i <= n ; i ++ ) log[i] = log[i >> 1] + 1;
for(i = 1 ; i <= log[n] ; i ++ )
{
for(j = 1 ; j <= n - (1 << i) + 1 ; j ++ )
{
if(maxn[j][i - 1] > maxn[j + (1 << (i - 1))][i - 1]) maxn[j][i] = maxn[j][i - 1] , maxp[j][i] = maxp[j][i - 1];
else maxn[j][i] = maxn[j + (1 << (i - 1))][i - 1] , maxp[j][i] = maxp[j + (1 << (i - 1))][i - 1];
}
}
for(i = 1 ; i <= n - l + 1 ; i ++ ) x = i + l - 1 , y = min(i + r - 1 , n) , q.push(data(i , x , y , getmaxn(x , y) - sum[i - 1] , getmaxp(x , y)));
while(k -- )
{
tmp = q.top() , q.pop() , ans += tmp.v;
if(tmp.p > tmp.b) x = tmp.b , y = tmp.p - 1 , q.push(data(tmp.t , x , y , getmaxn(x , y) - sum[tmp.t - 1] , getmaxp(x , y)));
if(tmp.p < tmp.e) x = tmp.p + 1 , y = tmp.e , q.push(data(tmp.t , x , y , getmaxn(x , y) - sum[tmp.t - 1] , getmaxp(x , y)));
}
printf("%lld\n" , ans);
return 0;
}
【bzoj2006】[NOI2010]超级钢琴 倍增RMQ+STL-堆的更多相关文章
- [NOI2010]超级钢琴 倍增
[NOI2010]超级钢琴 倍增 题面 暴力:枚举区间丢入堆\(O(n^2logn)\) 正解:考虑每次枚举和弦起点\(s\),那么以\(s\)为起点的和弦为\(sum[t]-sum[s](s+L-1 ...
- bzoj2006 [NOI2010]超级钢琴 (及其拓展)
bzoj2006 [NOI2010]超级钢琴 给定一个序列,求长度在 \([L,\ R]\) 之间的区间和的前 \(k\) 大之和 \(n\leq5\times10^5,\ k\leq2\times1 ...
- BZOJ2006 [NOI2010]超级钢琴 【堆 + RMQ】
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MB Submit: 3446 Solved: 1692 [Submit][Sta ...
- P2048 [NOI2010]超级钢琴(RMQ+堆+贪心)
P2048 [NOI2010]超级钢琴 区间和--->前缀和做差 多次查询区间和最大--->前缀和RMQ 每次取出最大的区间和--->堆 于是我们设个3元组$(o,l,r)$,表示左 ...
- [BZOJ2006][NOI2010]超级钢琴(ST表+堆)
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 3679 Solved: 1828[Submit][Statu ...
- 【BZOJ 2006】2006: [NOI2010]超级钢琴(RMQ+优先队列)
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MBSubmit: 2792 Solved: 1388 Description 小 ...
- bzoj千题计划162:bzoj2006: [NOI2010]超级钢琴
http://www.lydsy.com/JudgeOnline/problem.php?id=2006 输出最大的k个 sum[r]-sum[l-1] (L<=r-l+1<=R) 之和 ...
- [BZOJ2006] [NOI2010]超级钢琴 主席树+贪心+优先队列
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MBSubmit: 3591 Solved: 1780[Submit][Statu ...
- [NOI2010]超级钢琴(RMQ+堆)
小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为1至n.第i个音符的美妙度为Ai,其中Ai可正可负 ...
随机推荐
- cent os 上安装 matlab
下载和安装可以参考,这个链接: https://lanseyujie.com/post/matlab-download-and-activate.html 上面这链接,在创建桌面快捷键时,未能创建,c ...
- Angular-chart.js 使用说明(基于angular.js工程)
Angular-chart.js是基于Chart.js的angular组件,引入项目后直接操作数据即可. 引用方法: 分别将Chart.js.angular-chart.js.angular-c ...
- (转)IP地址分配原理
网络模型介绍 在计算机网络中有著名的OSI七层协议体系结构,概念清楚,理论完整,但是它既复杂又不实用.TCP/IP体系结构则不同,得到的广泛的应用.最终结合OSI和TCP/IP的优点,采用了一种只有五 ...
- Yaf学习(三)----Yaf类库Library和Model的命名规则
1.Yaf的library和model的文件命名规则和调用 1.1在项目中,往往需要封装一些,如redis,不同的产品需要用不同的库等等等,这就涉及到封装 1.在 Yaf 中,我们可以写一个单例模式的 ...
- haproxy + keepalived 实现高可用负载均衡集群
1. 首先准备两台tomcat机器,作为集群的单点server. 第一台: 1)tomcat,需要Java的支持,所以同样要安装Java环境. 安装非常简单. tar xf jdk-7u65-lin ...
- Leecode刷题之旅-C语言/python-100相同的树
/* * @lc app=leetcode.cn id=100 lang=c * * [100] 相同的树 * * https://leetcode-cn.com/problems/same-tree ...
- Leecode刷题之旅-C语言/python-35.搜索插入位置
/* * @lc app=leetcode.cn id=35 lang=c * * [35] 搜索插入位置 * * https://leetcode-cn.com/problems/search-in ...
- uva 509 RAID!(磁盘数据)
来自 https://blog.csdn.net/su_cicada/article/details/80085318 习题4-7 RAID技术(RAID!, ACM/ICPC World Final ...
- Kubernetes-深入分析集群安全机制
Kubernetes过一系列机制来实现集群的安全机制,包括API Server的认证授权.准入控制机制及保护敏感信息的Secret机制等.集群的安全性必须考虑以下的几个目标: 保证容器与其所在宿主机的 ...
- 接口和lambda表达式笔记
接口 接口是双方,即服务提供方和想让它们的对象对服务是可用的那些类,之间约定的一种机制. 声明一个接口 public interface IntSequence{ //不提供实现,则该方法为抽象方法, ...