BZOJ 2006: [NOI2010]超级钢琴
2006: [NOI2010]超级钢琴
Time Limit: 20 Sec Memory Limit: 552 MB
Submit: 2613 Solved: 1297
[Submit][Status][Discuss]
Description
Input
Output
只有一个整数,表示乐曲美妙度的最大值。
Sample Input
3
2
-6
8
Sample Output
【样例说明】
共有5种不同的超级和弦:
音符1 ~ 2,美妙度为3 + 2 = 5
音符2 ~ 3,美妙度为2 + (-6) = -4
音符3 ~ 4,美妙度为(-6) + 8 = 2
音符1 ~ 3,美妙度为3 + 2 + (-6) = -1
音符2 ~ 4,美妙度为2 + (-6) + 8 = 4
最优方案为:乐曲由和弦1,和弦3,和弦5组成,美妙度为5 + 2 + 4 = 11。
HINT
Source
学习的题解,转载在下——
堆+RMQ
定义一个四元组(i,l,r,t)表示是以第i个为开头的,结尾在l-r之间(满足长度>=L,<=R),且结尾在t的权值和最大。
首先把以每一位开头的四元组加入堆,此时的l,r是恰好满足长度>=L,<=R;
堆中排序的关键字是权值之和。
然后把堆顶取出(i,l,r,t),然后再把(i,l,t-1,t')和(i,t+1,r,t'')加入堆中(因为要满足任意两个序列不同的条件)
那么四元组中的t如何快速求出?
用RMQ即可:维护前缀和,权值和就是sum[t]-sum[i-1],对于用一个四元组sum[i-1]相同,因此只要找到l-r中sum[i]最大的即可。
- #include <queue>
- #include <cstdio>
- inline int nextChar(void)
- {
- const static int siz = ;
- static char buf[siz];
- static char *hd = buf + siz;
- static char *tl = buf + siz;
- if (hd == tl)
- fread(hd = buf, , siz, stdin);
- return *hd++;
- }
- inline int nextInt(void)
- {
- register int ret = ;
- register int neg = false;
- register int bit = nextChar();
- for (; bit < ; bit = nextChar())
- if (bit == '-')neg ^= true;
- for (; bit > ; bit = nextChar())
- ret = ret * + bit - ;
- return neg ? -ret : ret;
- }
- const int siz = ;
- int N, M, L, R;
- int num[siz];
- int sum[siz];
- int st[siz][], log[siz];
- inline void preworkRMQ(void)
- {
- for (int i = ; i <= N; ++i)
- st[i][] = i;
- for (int i = ; ( << i) <= N; ++i)
- for (int j = ; j + ( << i) - <= N; ++j)
- {
- int x = st[j][i - ];
- int y = st[j + ( << (i - ))][i - ];
- st[j][i] = sum[x] > sum[y] ? x : y;
- }
- log[] = -;
- for (int i = ; i <= N; ++i)
- log[i] = log[i >> ] + ;
- }
- inline int query(int l, int r)
- {
- if (l == r)
- return l;
- int t = log[r - l + ];
- int x = st[l][t];
- int y = st[r - ( << t) + ][t];
- return sum[x] > sum[y] ? x : y;
- }
- struct data
- {
- int p, t, l, r;
- data(void) {};
- data(int a, int b, int c, int d) :
- p(a), t(b), l(c), r(d) {};
- };
- inline bool operator < (const data &a, const data &b)
- {
- return sum[a.t] - sum[a.p - ] < sum[b.t] - sum[b.p - ];
- }
- std::priority_queue<data> h;
- signed main(void)
- {
- N = nextInt();
- M = nextInt();
- L = nextInt();
- R = nextInt();
- for (int i = ; i <= N; ++i)
- num[i] = nextInt();
- for (int i = ; i <= N; ++i)
- sum[i] = sum[i - ] + num[i];
- preworkRMQ();
- long long ans = ;
- for (int i = ; i <= N - L + ; ++i)
- {
- int l = i + L - ;
- int r = i + R - ;
- if (r > N)
- r = N;
- h.push(data(i, query(l, r), l, r));
- }
- for (int i = ; i <= M; ++i)
- {
- data top = h.top(); h.pop();
- ans += sum[top.t] - sum[top.p - ];
- if (top.t > top.l)
- h.push(data(top.p, query(top.l, top.t - ), top.l, top.t - ));
- if (top.t < top.r)
- h.push(data(top.p, query(top.t + , top.r), top.t + , top.r));
- }
- printf("%lld\n", ans);
- }
@Author: YouSiki
BZOJ 2006: [NOI2010]超级钢琴的更多相关文章
- Bzoj 2006: [NOI2010]超级钢琴 堆,ST表
2006: [NOI2010]超级钢琴 Time Limit: 20 Sec Memory Limit: 552 MBSubmit: 2222 Solved: 1082[Submit][Statu ...
- BZOJ 2006: [NOI2010]超级钢琴( RMQ + 堆 )
取最大的K个, 用堆和RMQ来加速... ----------------------------------------------------------------- #include<c ...
- 洛谷 P2048 BZOJ 2006 [NOI2010]超级钢琴
题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为1至n.第i个音符的美妙度为Ai,其中A ...
- BZOJ.2006.[NOI2010]超级钢琴(贪心 堆)
BZOJ 洛谷 思路和BZOJ3784一样,用前缀和+堆维护.做那题吧,不赘述啦. (没错我就是水一个AC) //54620kb 1060ms #include <queue> #incl ...
- bzoj 2006 [NOI2010]超级钢琴——ST表+堆
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2006 每个右端点的左端点在一个区间内:用堆记录端点位置.可选区间,按价值排序:拿出一个后也许 ...
- BZOJ 2006 [NOI2010]超级钢琴 (堆+主席树)
题面:BZOJ传送门 洛谷传送门 让你求前$K$大的子序列和,$n\leq 5*10^{5}$ 只想到了个$nlog^{2}n$的做法,似乎要被卡常就看题解了.. 好神奇的操作啊,我傻了 我们把序列和 ...
- BZOJ 2006 NOI2010 超级钢琴 划分树+堆
题目大意:给定一个序列.找到k个长度在[l,r]之间的序列.使得和最大 暴力O(n^2logn),肯定过不去 看到这题的第一眼我OTZ了一下午... 后来研究了非常久别人的题解才弄明确怎么回事...蒟 ...
- BZOJ 2006: [NOI2010]超级钢琴 [ST表+堆 | 主席树]
题意: 一个序列,求k个不相同的长度属于\([L,R]\)的区间使得和最大 前缀和,对于每个r找最小的a[l] 然后我yy了一个可持久化线段树做法...也许会T 实际上主席树就可以了,区间k小值 然后 ...
- bzoj 2006: [NOI2010]超级钢琴【st表+堆】
设计一个五元组(i,l,r,p,v),表示在以i为左端点,右端点落在(l,r)中的情况下,取最大值v时右端点落在p.把这个五元组塞到优先队列里,以v排序,每次取出一个,然后把这个取过的五元组分成两个( ...
随机推荐
- gearman 安装
yum install gperfyum install libevent-develyum install libuuid-develwget https://launchpad.net/gearm ...
- Canvas的width,height 和 样式中Canvas的width,height
控制Canvas的大小,有两种方式: 1:直接设置Canvas标签上的书width,height属性值; 2:通过Css设置Canvas的width,height; 这两种方式,区别是很大的. 1:C ...
- ipython notebook 浏览器中编写数学公式和现实
Python Notebook简介1 http://www.cnblogs.com/cbscan/p/3545084.html $ python -m IPython http://pypi.pyth ...
- React Native学习笔记之2
1:如何创建一个react native工程 首先进入到指定文件夹里面,然后在终端执行react-native init ReactNativeProject :其中ReactNativeProjec ...
- ERR_CONTENT_DECODING_FAILED错误的原因和解决办法
1. ERR_CONTENT_DECODING_FAILED错误的原因 这种错误通常发生于Http请求中的头部信息标识内容是gzip编码的,但实际上不是. 2. ERR_CONTENT_DECODIN ...
- CANopen学习——感性认知
看不懂的知识硬着头皮也要看.读了当时虽然不理解,但脑子里对其相关名词.概念有印象,继续看下去,多读几遍,一定会在某个地方顿悟. CAN总线只是定义了物理层和数据链路层,并没有定义应用层.这么优秀的总 ...
- Laravel中的ajax跨域请求
最近接触Laravel框架ajax跨域请求的过程中遇到一些问题,在这里做下总结. 一开始发起ajax请求一直报500错误,搜索相关资料后发现Laravel要允许跨域请求可以加入Cors中间件,代码如下 ...
- Cookie和Session的那些事儿
Cookie和Session都是为了保持用户的访问状态,一方面为了方便业务实现,另一方面为了简化服务端的程序设计,提高访问性能.Cookie是客户端(也就是浏览器端)的技术,设置了Cookie之后,每 ...
- [Network Analysis] 复杂网络分析总结
在我们的现实生活中,许多复杂系统都可以建模成一种复杂网络进行分析,比如常见的电力网络.航空网络.交通网络.计算机网络以及社交网络等等.复杂网络不仅是一种数据的表现形式,它同样也是一种科学研究的手段.复 ...
- [LeetCode] Evaluate Reverse Polish Notation 计算逆波兰表达式
Evaluate the value of an arithmetic expression in Reverse Polish Notation. Valid operators are +, -, ...