洛谷题目链接:[NOI2010]超级钢琴

题目描述

小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐。

这架超级钢琴可以弹奏出n个音符,编号为1至n。第i个音符的美妙度为Ai,其中Ai可正可负。

一个“超级和弦”由若干个编号连续的音符组成,包含的音符个数不少于L且不多于R。我们定义超级和弦的美妙度为其包含的所有音符的美妙度之和。两个超级和弦被认为是相同的,当且仅当这两个超级和弦所包含的音符集合是相同的。

小Z决定创作一首由k个超级和弦组成的乐曲,为了使得乐曲更加动听,小Z要求该乐曲由k个不同的超级和弦组成。我们定义一首乐曲的美妙度为其所包含的所有超级和弦的美妙度之和。小Z想知道他能够创作出来的乐曲美妙度最大值是多少。

输入输出格式

输入格式:

输入第一行包含四个正整数n, k, L, R。其中n为音符的个数,k为乐曲所包含的超级和弦个数,L和R分别是超级和弦所包含音符个数的下限和上限。

接下来n行,每行包含一个整数Ai,表示按编号从小到大每个音符的美妙度。

输出格式:

输出只有一个整数,表示乐曲美妙度的最大值。

输入输出样例

输入样例#1:

4 3 2 3

3

2

-6

8

输出样例#1:

11

说明

共有5种不同的超级和弦:

1.    音符1 ~ 2,美妙度为3 + 2 = 5
2. 音符2 ~ 3,美妙度为2 + (-6) = -4
3. 音符3 ~ 4,美妙度为(-6) + 8 = 2
4. 音符1 ~ 3,美妙度为3 + 2 + (-6) = -1
5. 音符2 ~ 4,美妙度为2 + (-6) + 8 = 4

最优方案为:乐曲由和弦1,和弦3,和弦5组成,美妙度为5 + 2 + 4 = 11。

所有数据满足:-1000 ≤ Ai ≤ 1000,1 ≤ L ≤ R ≤ n且保证一定存在满足要求的乐曲。


一句话题意: 在一个长度为\(n\)的序列中,要求出其中的最大的\(k\)段不同的子序列,并且满足这个子序列的长度∈\([l,r]\).


题解: 既然限定了区间的长度,所以可以考虑枚举区间的左端点,那么这样合法的右端点就组成了一个区间,对于同一个左端点,它的合法右端点的前缀和的最大值显然是这个左端点能取得的最大值,并且合法右端点的区间的第2大,第3大减去左端点的前缀和也一定是单调递减的.这让我们联想到[洛谷P1631] 序列合并.

通过枚举左端点,在合法右端点中查找第1大的前缀和,然后将它减去左端点的前缀和,再将这个值加入堆中,这样每次取出堆中的最大值,加入答案,这样堆维护的事实上就是一个区间.它具有左端点,并且记录了查找到第几大,以这个区间的权值作为键值维护堆.

然后每次取出堆中的最大值,也就是取出了这个区间,然后就可以将这个左端点对应的合法右端点的前缀和的下一个最大值加入堆中,这样维护出的前\(k\)大一定是最优的.

静态区间第\(k\)大可以用主席树实现,其他的参数注意不要打错,要开long long.

#include<bits/stdc++.h>
using namespace std;
const int N=500000+5;
typedef int _int;
#define int long long int n, k, l, r, a[N], s[N], rk[N], root[N], temp[N], vis[N], size, cnt = 0, ans = 0; struct president_tree{
int ls, rs, cnt;
}t[N*20]; struct number{
int id1, id2, val;
bool operator < (const number &a) const{
return val < a.val;
}
}; priority_queue <number> h; inline int gi(){
int ans = 0, f = 1; char i = getchar();
while(i<'0' || i>'9'){ if(i == '-') f = -1; i = getchar(); }
while(i>='0' && i<='9') ans = ans*10+i-'0', i = getchar();
return ans * f;
} inline void update(int &x, int last, int pos, int l = 1, int r = size){
x = ++cnt; t[x] = t[last]; t[x].cnt++;
if(l == r) return; int mid = (l+r>>1);
if(pos <= mid) update(t[x].ls, t[last].ls, pos, l, mid);
else update(t[x].rs, t[last].rs, pos, mid+1, r);
} inline int query(int x, int last, int k, int l = 1, int r = size){
if(l == r) return vis[l];
int mid = (l+r>>1), sum = t[t[x].ls].cnt-t[t[last].ls].cnt;
if(k <= sum) return query(t[x].ls, t[last].ls, k, l, mid);
return query(t[x].rs, t[last].rs, k-sum, mid+1, r);
} _int main(){
// freopen("sequence.in", "r", stdin);
// freopen("sequence.out", "w", stdout);
n = gi(), k = gi(), l = gi(), r = gi();
for(int i=1;i<=n;i++) a[i] = gi(), s[i] = s[i-1]+a[i];
memcpy(temp, s, sizeof(s));
sort(temp+1, temp+n+1); size = unique(temp+1, temp+n+1)-temp-1;
for(int i=1;i<=n;i++){
rk[i] = lower_bound(temp+1, temp+size+1, s[i])-temp;
vis[rk[i]] = s[i];
}
for(int i=1;i<=n;i++) update(root[i], root[i-1], rk[i]);
for(int tmp, i=1;i<=n-l+1;i++){
int lim = min(n, i+r-1);
h.push((number){ i, lim-i-l+2, query(root[lim], root[i+l-2], lim-i-l+2)-s[i-1] });
}
for(int i=1;i<=k;i++){
number top, tmp; top = h.top(); h.pop();
ans += top.val;
if(top.id2 <= 1) continue;
tmp.id2 = top.id2-1, tmp.id1 = top.id1;
int lim = min(n, top.id1+r-1);
tmp.val = query(root[lim], root[tmp.id1+l-2], tmp.id2)-s[top.id1-1];
h.push(tmp);
}
cout << ans << endl;
return 0;
}

[洛谷P2048] [NOI2010] 超级钢琴的更多相关文章

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

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

  2. 洛谷 P2048 [NOI2010]超级钢琴 || Fantasy

    https://www.luogu.org/problemnew/show/P2048 http://www.lydsy.com/JudgeOnline/problem.php?id=2006 首先计 ...

  3. 洛谷P2048 [NOI2010]超级钢琴 题解

    2019/11/14 更新日志: 近期发现这篇题解有点烂,更新一下,删繁就简,详细重点.代码多加了注释.就酱紫啦! 正解步骤 我们需要先算美妙度的前缀和,并初始化RMQ. 循环 \(i\) 从 \(1 ...

  4. 洛谷 P2048 [NOI2010]超级钢琴(优先队列,RMQ)

    传送门 我们定义$(p,l,r)=max\{sum[t]-sum[p-1],p+l-1\leq t\leq p+r-1 \}$ 那么因为对每一个$p$来说$sum[p-1]$是一个定值,所以我们只要在 ...

  5. 洛谷P0248 [NOI2010] 超级钢琴 [RMQ,贪心]

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

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

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

  7. 【题解】P2048 [NOI2010]超级钢琴

    [题解][P2048 NOI2010]超级钢琴 一道非常套路的题目.是堆的套路题. 考虑前缀和,我们要是确定了左端点,就只需要在右端区间查询最大的那个加进来就好了.\(sum_j-sum_{i-1}​ ...

  8. LGOJ P2048 [NOI2010]超级钢琴

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

  9. 洛谷$P$2048 超级钢琴 $[NOI2010]$ $rmq$/主席树

    正解:$rmq$/主席树 解题报告: 传送门! 好像看过这题挺多次辣,,,$QwQ$ 之前$NOIp$的时候$cjk$学长讲课讲了这题(虽然那时候的$gql$太菜辣并麻油落实这道_(:з」∠)_,然后 ...

随机推荐

  1. selenium实现文件上传方法汇总(AutoIt、win32GUI、sengkeys)---基于python

    在使用selenium进行UI自动化测试时,经常会遇到一个关于本地文件上传的问题,解决此问题一般分两种情况: 1. 元素标签为input 2.非input型上传 下面我们分别对着两种情况进行实例分析 ...

  2. SGU 326 Perspective(最大流)

    Description Breaking news! A Russian billionaire has bought a yet undisclosed NBA team. He's plannin ...

  3. 关于14道魔鬼js考题的整理

    1.(function(){ return typeof arguments })(); 这里返回时是argument类型,它是个类数组,也就对象,所以是object,准确谁是[object argu ...

  4. Python学习之路4 - 文件操作&编码转换

    文件操作 文件操作大概分三步: 把文件打开. 操作文件. 把文件关上. 打开文件 打开文件用open()函数,打开成功后返回一个资源,具体语法如下. open(要打开的文件,打开方式,打开文件的格式, ...

  5. 浅述Try {} Catch{} 作用

    using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace Test ...

  6. Jira & SVN & Chrome extensions

    Jira & SVN & Chrome extensions Plugins SVN & Jira Plugins ok selector bug document.query ...

  7. mysql(一) 关联查询的方式

    mysql做关联查询时,一般使用join....on.....的语法. 但还有其它两种语法形式,三者的主要区别在于书写形式,其余方面并无太多差异. 如下三种形式: select * from trad ...

  8. placeholder 颜色

    /* placeholder颜色 */::-webkit-input-placeholder { /* WebKit browsers */color: #ccc;}:-moz-placeholder ...

  9. hdu3625-Rooms

    题目 有\(n\)个房间,\(n\)个钥匙,每个钥匙随机出现在一个房间里,一个房间里有且仅有一个钥匙.我们现在手上没有钥匙,但我们要搜索所有的房间,所以我们有\(k\)次机会把一个房间炸开.一号房间里 ...

  10. FZU 1492 地震预测(链表)

    实际上把数组排序一遍加入链表中,再记录好数组原来的数在链表中的位置.我们只需要维护链表的删除操作就可以了. # include <cstdio> # include <cstring ...