Description

​ 给定一个长度为n的区间,询问前k大的区间和,区间长度\(\in [L, R]\)。 $ n, k <= 500000$

Solution

​ 首先求前缀和。把一个区间拆成两个前缀和之差。

​ 对于一个固定的右端点。我们首先需要最小化左端点。那么我们就在题目条件允许的范围内先求出左端点最小的位置, 但是一个区间的次大值可能超过别的区间的最大值, 这很麻烦。

​ 开一个堆, 维护一个三元组(Pos, Kth, Val), 表示以Pos为右端点,本次查到区间第Kth大,值为Val。

​ 然后把一个区间从堆中取出来时, 就把他的(Pos, Kth + 1, Val_)加入堆。

​ 然后只要取k次即可。

​ 时间复杂度\(O((n + k)log_{2}^{n})\)

反思

​ 刚刚开始想到把每个右端点的左端点按照大小都加进小根堆, 小根堆大小为K,如果超过就弹出一个。 如果当前值小于堆顶就break, 然后就被卡T到8s

​ 最后发现,其实你并不需要把全部的都加入堆,因为只要当前的区间被弹出,才有可能用他的次大值。

​ 所以我们只要贪心取最大值就可以的。

#include<bits/stdc++.h>
using namespace std;
#define rep(i, a, b) for(int i = (a), i##_end_ = (b); i <= i##_end_; ++i)
#define drep(i, a, b) for(int i = (a), i##_end_ = (b); i >= i##_end_; --i)
#define clar(a, b) memset((a), (b), sizeof(a))
#define debug(...) fprintf(stderr, __VA_ARGS__)
#define Debug(s) debug("The massage in line %d, Function %s: %s\n", __LINE__, __FUNCTION__, s)
typedef long long LL;
typedef long double LD;
const int BUF_SIZE = (int)1e6 + 10;
struct fastIO {
char buf[BUF_SIZE], buf1[BUF_SIZE];
int cur, cur1;
FILE *in, *out;
fastIO() {
cur = BUF_SIZE, in = stdin, out = stdout;
cur1 = 0;
}
inline char getchar() {
if(cur == BUF_SIZE) fread(buf, BUF_SIZE, 1, in), cur = 0;
return *(buf + (cur++));
}
inline void putchar(char ch) {
*(buf1 + (cur1++)) = ch;
if (cur1 == BUF_SIZE) fwrite(buf1, BUF_SIZE, 1, out), cur1 = 0;
}
inline int flush() {
if (cur1 > 0) fwrite(buf1, cur1, 1, out);
return cur1 = 0;
}
}IO;
#define getchar IO.getchar
#define putchar IO.putchar
int read() {
char ch = getchar();
int x = 0, flag = 1;
for(;!isdigit(ch); ch = getchar()) if(ch == '-') flag *= -1;
for(;isdigit(ch); ch = getchar()) x = x * 10 + ch - 48;
return x * flag;
}
void write(int x) {
if(x < 0) putchar('-'), x = -x;
if(x >= 10) write(x / 10);
putchar(x % 10 + 48);
}
void putString(char s[], char EndChar = '\n') {
rep(i, 0, strlen(s) - 1) putchar(*(s + i));
if(~EndChar) putchar(EndChar);
} #define Maxn 500009
int n, k, L, R, a[Maxn];
LL s[Maxn];
template <typename T, int (*Cmp)(T a, T b)> struct Heap {
T a[Maxn];
int len;
void push(T val) {
a[++len] = val;
push_heap(a + 1, a + len + 1, Cmp);
}
T pop() { pop_heap(a + 1, a + len + 1, Cmp); return a[len--];}
T top() {return a[1];}
T empty() { return !len;}
};
LL W[Maxn], d, Ns[Maxn];
namespace Chairman {
int t[Maxn * 64], lc[Maxn * 64], rc[Maxn * 64], amt, rt[Maxn];
int build(int l, int r) {
int u = ++amt, mid = (l + r) >> 1;
if(l == r) return u;
lc[u] = build(l, mid);
rc[u] = build(mid + 1, r);
return u;
}
int modify(int rt, int l, int r, int pos) {
int mid = (l + r) >> 1, u = ++amt;
t[u] = t[rt] + 1; lc[u] = lc[rt], rc[u] = rc[rt];
if(l == r) return u;
(pos <= mid) ? (lc[u] = modify(lc[rt], l, mid, pos)) : (rc[u] = modify(rc[rt], mid + 1, r, pos));
return u;
}
int query(int rt0, int rt1, int l, int r, int pos) {
if(l == r) return l;
/**/ int res = t[lc[rt1]] - t[lc[rt0]], mid = (l + r) >> 1;
return (res >= pos) ? (query(lc[rt0], lc[rt1], l, mid, pos)) : (query(rc[rt0], rc[rt1], mid + 1, r, pos - res));
}
}
namespace INIT {
void Main() {
n = read(), k = read(); L = read(), R = read();
rep(i, 1, n) a[i] = read(), W[i + 1] = (s[i] = s[i - 1] + a[i]);
sort(W + 1, W + n + 2);
d = unique(W + 1, W + n + 2) - W - 1;
rep(i, 0, n) Ns[i] = lower_bound(W + 1, W + d + 1, s[i]) - W;
}
}
namespace SOLVE {
using namespace Chairman;
struct node {
int pos, kth;
LL val;
};
int cmp(node tarA, node tarB) { return tarA.val < tarB.val;}
Heap <node, cmp> hep;
void Main() {
rt[0] = build(1, d), rt[1] = modify(rt[0], 1, d, Ns[0]);
rep(i, 1, n) rt[i + 1] = modify(rt[i], 1, d, Ns[i]);
rep(i, 1, n) {
int bound1 = max(i - R, 0), bound2 = i - L;
if(bound2 < bound1) continue;
int cur = query(rt[bound1], rt[bound2 + 1], 1, d, 1);
hep.push((node){i, 1, s[i] - W[cur]});
}
LL ans = 0;
rep(i, 1, k) {
node Tmp = hep.pop();
ans += Tmp.val;
int bound1 = max(Tmp.pos - R, 0), bound2 = Tmp.pos - L;
if(Tmp.kth + 1 > bound2 - bound1 + 1) continue;
int cur = query(rt[bound1], rt[bound2 + 1], 1, d, Tmp.kth + 1);
hep.push((node){Tmp.pos, Tmp.kth + 1, s[Tmp.pos] - W[cur]});
}
cout << ans << endl;
}
}
int main() {
#ifdef Qrsikno
freopen("BZOJ2006.in", "r", stdin);
freopen("BZOJ2006.out", "w", stdout);
#endif
INIT :: Main();
SOLVE :: Main();
#ifdef Qrsikno
debug("\nRunning time: %.3lf(s)\n", clock() * 1.0 / CLOCKS_PER_SEC);
#endif
return IO.flush();
}

BZOJ2006 超级钢琴的更多相关文章

  1. [NOI2010][bzoj2006] 超级钢琴 [主席树/ST表+堆]

    题面: 传送门 思路: 首先容易想到用堆维护的O(n2logn)暴力 那么肯定就是在这个基础上套数据结构了[愉快] 然而我因为过于蒟蒻......只想得到主席树暴力***过去的方法 大概就是把前缀和算 ...

  2. bzoj2006 noi2010 超级钢琴 主席树 + 优先队列

    Time Limit: 20 Sec  Memory Limit: 552 MBSubmit: 2435  Solved: 1195 Description 小 Z是一个小有名气的钢琴家,最近C博士送 ...

  3. 【BZOJ2006】超级钢琴(主席树,优先队列)

    [BZOJ2006]超级钢琴(主席树,优先队列) 题面 BZOJ 题解 既然是一段区间 首先就要变成单点 所以求一个前缀和 这个时候贪心很明显了: 枚举每一个点和可以和它组成一段的可行的点 全部丢进一 ...

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

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

  5. 【BZOJ2006】【NOI2010】超级钢琴(主席树,优先队列)

    [BZOJ2006]超级钢琴(主席树,优先队列) 题面 BZOJ 题解 既然是一段区间 首先就要变成单点 所以求一个前缀和 这个时候贪心很明显了: 枚举每一个点和可以和它组成一段的可行的点 全部丢进一 ...

  6. 【BZOJ2006】[NOI2010]超级钢琴 ST表+堆

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

  7. 【BZOJ-2006】超级钢琴 ST表 + 堆 (一类经典问题)

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

  8. BZOJ2006[NOI2010]超级钢琴——堆+主席树

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

  9. bzoj千题计划162:bzoj2006: [NOI2010]超级钢琴

    http://www.lydsy.com/JudgeOnline/problem.php?id=2006 输出最大的k个 sum[r]-sum[l-1] (L<=r-l+1<=R) 之和 ...

随机推荐

  1. Tomcat+Servlet登录页面实例

    概念   Tomcat server是一个免费的开放源码的Web 应用server,属于轻量级应用server,在中小型系统和并发訪问用户不是非常多的场合下被普遍使用,是开发和调试JSP 程序的首选. ...

  2. 【转载】Unix设计哲学 & 回车换行八卦 & EOF八卦 & UNIX目录结构八卦

    昨天看了这篇文章 <关于Unix哲学> 首先用了两个例子,用风扇吹出空肥皂盒 和 太空铅笔,来说明简单设计也能派上作用吧. Unix哲学,Wikipedia上列出了好几个版本,不同的人有不 ...

  3. simple-todo: 一个简易的 todo 程序 - django版

    今天无意间看到  simple-todo: 一个简易的 todo 程序 - web.py 中文教程 ,然后发现竟然有好多的版本 http://simple-is-better.com/news/tag ...

  4. Java基础面试:集合、内部类、线程

    package test; import java.util.Hashtable; import java.util.Map; public class test { public static St ...

  5. linux安装anaconda中的问题及解决办法

    安装过程: 0:在ananconda官网网站上下载anaconda的linux版本https://www.anaconda.com/download/: 1:linux上切换到下载目录后(用cd), ...

  6. 【iOS系列】-UITableView的使用

    UITableView的使用: 第一:数据展示条件 1,UITableView的所有数据都是由数据源(dataSource)提供,所以想在UITableView展示数据,必须设置UITableview ...

  7. Spring简单实现数据源的动态切换

    Spring简单实现数据源的动态切换: 1. 创建一个数据源切换类: 2. 继承AbstractRoutingDataSource,创建多数据源路由类,并注入到spring的配置文件中: 3. AOP ...

  8. 嵌入式Linux内核+根文件系统构建工具-Buildroot 快速入手指导【转】

    本文转载自:https://my.oschina.net/freeblues/blog/596448 嵌入式Linux内核+根文件系统构建工具-Buildroot 快速入手指导 buildroot 是 ...

  9. caioj1272&&codeforces 148D: 概率期望值3:抓老鼠

    这道真的是好题,不卡精度,不卡细节,但是思考的方式很巧妙! 一开始大家跟我想的应该差不多,用f[i][j]表示有i只白老鼠,j只黑老鼠的胜率,然后跑DP,然后我就发现,这样怎么做?还有一种不胜不负的平 ...

  10. HDU3642 Get The Treasury —— 求矩形交体积 线段树 + 扫描线 + 离散化

    题目链接:https://vjudge.net/problem/HDU-3642 Jack knows that there is a great underground treasury in a ...