题目

一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整。给你一个

长度为n的序列s。回答Q个这样的询问:s的左端点在[a,b]之间,右端点在[c,d]之间的子序列中,最大的中位数。

其中a<b<c<d。位置也从0开始标号。我会使用一些方式强制你在线。

输入格式

第一行序列长度n。接下来n行按顺序给出a中的数。

接下来一行Q。然后Q行每行a,b,c,d,我们令上个询问的答案是

x(如果这是第一个询问则x=0)。

令数组q={(a+x)%n,(b+x)%n,(c+x)%n,(d+x)%n}。

将q从小到大排序之后,令真正的

要询问的a=q[0],b=q[1],c=q[2],d=q[3]。  

输入保证满足条件。

第一行所谓“排过序”指的是从大到小排序!

输出格式

Q行依次给出询问的答案。

输入样例

5

170337785

271451044

22430280

969056313

206452321

3

3 1 0 2

2 3 1 4

3 1 4 0

输出样例

271451044

271451044

969056313

提示

  0:n,Q<=100

1,...,5:n<=2000

0,...,19:n<=20000,Q<=25000

题解

首先我们理解一下题意:

从0开始的序列降序排序,取b[n / 2]向下取整为中位数,实际上就是一个往大取的中位数

比如说有偶数个【比如6个】,那么中位数就是第3大而不是第4大

奇数个自然就是取中位数,这些模拟一下就可以推出

怎么求?

假设我们指定一个数x,将大于等于x的设为1,小于x的设为-1

如果一个区间和非负,那么这个区间的中位数至少为x,因为比x大的个数不少于比x小的个数

可以发现这样的x具有单调性,可以二分x

所以我们只要二分x,检验选取左右端点能取出的最大区间和是否非负

现在问题就转化成了:

对每个x,建立一个1,-1序列,并求出最大区间和

当x增大时序列只会有一个元素改变,而又需要维护动态信息

很自然可以想到主席树

所以我们将原序列排个序,按顺序建树,二分 + 主席树就可以解决了

检验左区间最大后缀和 + 中间区间和 + 右区间最大前缀和 是否非负,如果是就可行

否则偏大

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define LL long long int
#define REP(i,n) for (int i = 1; i <= (n); i++)
using namespace std;
const int maxn = 20005,maxm = 3000005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57) {out = (out << 3) + (out << 1) + c - '0'; c = getchar();}
return out * flag;
}
int ms[maxm],ml[maxm],mr[maxm],sum[maxm],ls[maxm],rs[maxm],cnt,rt[maxn];
int id[maxn],A[maxn],tot = 1,n,Q,q[6],lans;
inline bool cmp(const int& a,const int& b){return A[a] < A[b];}
void copy(int u,int pre){
ms[u] = ms[pre]; ml[u] = ml[pre]; mr[u] = mr[pre];
ls[u] = ls[pre]; rs[u] = rs[pre]; sum[u] = sum[pre];
}
void upd(int u){
sum[u] = sum[ls[u]] + sum[rs[u]];
ms[u] = max(max(ms[ls[u]],ms[rs[u]]),mr[ls[u]] + ml[rs[u]]);
ml[u] = max(ml[ls[u]],sum[ls[u]] + ml[rs[u]]);
mr[u] = max(mr[rs[u]],sum[rs[u]] + mr[ls[u]]);
}
void build(int& u,int l,int r){
u = ++cnt;
if (l == r){
sum[u] = ms[u] = ml[u] = mr[u] = 1;
return;
}
int mid = l + r >> 1;
build(ls[u],l,mid);
build(rs[u],mid + 1,r);
upd(u);
}
void modify(int& u,int pre,int l,int r,int pos,int v){
u = ++cnt; copy(u,pre);
if (l == r){sum[u] = v; ms[u] = ml[u] = mr[u] = v; return;}
int mid = l + r >> 1;
if (mid >= pos) modify(ls[u],ls[pre],l,mid,pos,v);
else modify(rs[u],rs[pre],mid + 1,r,pos,v);
upd(u);
}
struct node{int l,r,s,v;};
node query(int u,int l,int r,int L,int R){
if (l >= L && r <= R) return (node){ml[u],mr[u],ms[u],sum[u]};
int mid = l + r >> 1;
if (mid >= R) return query(ls[u],l,mid,L,R);
else if (mid < L) return query(rs[u],mid + 1,r,L,R);
else {
node a = query(ls[u],l,mid,L,R),b = query(rs[u],mid + 1,r,L,R);
return (node){max(a.l,a.v + b.l),max(b.r,b.v + a.r),
max(max(a.s,b.s),a.r + b.l),a.v + b.v};
}
}
void solve(int x,int y,int xx,int yy){
int l = 1,r = n,mid,t;
while (l < r){
mid = l + r + 1 >> 1;
t = query(rt[mid],1,n,x,y).r
+ (y + 1 <= xx - 1 ? query(rt[mid],1,n,y + 1,xx - 1).v : 0)
+ query(rt[mid],1,n,xx,yy).l;
if (t >= 0) l = mid;
else r = mid - 1;
}
printf("%d\n",lans = A[id[l]]);
}
int main(){
n = read();
REP(i,n) A[i] = read(),id[i] = i;
sort(id + 1,id + 1 + n,cmp);
build(rt[0],1,n);
for (int i = 1; i <= n; i++){
rt[i] = rt[i - 1];
if (i > 1) modify(rt[i],rt[i],1,n,id[i - 1],-1);
}
Q = read();
while (Q--){
for (int i = 0; i < 4; i++) q[i] = (read() + lans) % n;
sort(q,q + 4);
solve(q[0] + 1,q[1] + 1,q[2] + 1,q[3] + 1);
}
return 0;
}

BZOJ2653 middle 【二分 + 主席树】的更多相关文章

  1. [bzoj2653][middle] (二分 + 主席树)

    Description 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整. 给你一个长度为n的序列s. 回答Q个这样的询问:s的左端点在[a,b ...

  2. BZOJ2653 middle 【主席树】【二分】*

    BZOJ2653 middle Description 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你一个长度为n的序列s.回答Q个这样 ...

  3. 【BZOJ2653】Middle(主席树)

    [BZOJ2653]Middle(主席树) 题面 BZOJ 洛谷 Description 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你 ...

  4. 洛谷P4559 [JSOI2018]列队 【70分二分 + 主席树】

    题目链接 洛谷P4559 题解 只会做\(70\)分的\(O(nlog^2n)\) 如果本来就在区间内的人是不用动的,区间右边的人往区间最右的那些空位跑,区间左边的人往区间最左的那些空位跑 找到这些空 ...

  5. 【BZOJ 4556】[Tjoi2016&Heoi2016]字符串 SAM+二分+主席树

    这道题市面上就两种法:一种是SA+二分+主席树,一种是SAM+二分+主席树(有不少人打线段树合并???)(除此之外还有一种利用炒鸡水的数据的暴力SA,贼快.....)(当时学SA的时候没做这道题,现在 ...

  6. 【BZOJ2653】middle(主席树,二分)

    题意:一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整. 给你一个长度为n的序列s. 回答Q个这样的询问:s的左端点在[a,b]之间,右端点在[ ...

  7. 【洛谷2839/BZOJ2653】middle(主席树)

    题目: 洛谷2839 分析: 记\(s_i\)表示原序列中第\(i\)大的数. 考虑对于任意一个区间\([a,b]\),设它的中位数为\(s_m\),那么这个区间内大于等于\(s_m\)的数和小于\( ...

  8. 「BZOJ 2653」middle「主席树」「二分」

    题意 一个长度为\(n\)的序列\(a\),设其排过序之后为\(b\),其中位数定义为\(b[n/2]\),其中\(a,b\)从\(0\)开始标号,除法取下整.给你一个长度为\(n\)的序列\(s\) ...

  9. Codeforces Round #276 (Div. 1) E. Sign on Fence 二分+主席树

    E. Sign on Fence   Bizon the Champion has recently finished painting his wood fence. The fence consi ...

随机推荐

  1. UVA 1471 Defense Lines 防线 (LIS变形)

    给一个长度为n的序列,要求删除一个连续子序列,使剩下的序列有一个长度最大的连续递增子序列. 最简单的想法是枚举起点j和终点i,然后数一数,分别向前或向后能延伸的最长长度,记为g(i)和f(i).可以先 ...

  2. opencv中mat的type

    type表示了矩阵中元素的类型以及矩阵的通道个数,它是一系列的预定义的常量,其命名规则为CV_(位数)+(数据类型)+(通道数),由type()返回,但是返回值是int型,不是OpenCV预定义的宏( ...

  3. 判断是否是同一人的方法——equals()?在Person类中提供一个比较的方法compare()返回boolean值?对象自己和自己比?

    判断是否是同一人的方法——equals() 不能直接用per1==per2,这不是对象内容的比较而是存放对象地址的值得比较 在Person类中提供一个比较的方法compare()返回boolean值 ...

  4. C#:CodeSmith根据数据库中的表创建C#数据模型Model + 因为没有钱买正版,所以附加自己写的小代码

    对于C#面向对象的思想,我们习惯于将数据库中的表创建对应的数据模型: 但假如数据表很多时,我们手动增加模型类会显得很浪费时间: 这个时候有些人会用微软提供的EntityFrameWork,这个框架很强 ...

  5. 51nod——2502最多分成多少块

    数据范围好小... 题目中没说要升序降序,不过样例解释里可以看出是要升序. #include <bits/stdc++.h> using namespace std; ],b[],visi ...

  6. 转 Spring源码剖析——核心IOC容器原理

    Spring源码剖析——核心IOC容器原理 2016年08月05日 15:06:16 阅读数:8312 标签: spring源码ioc编程bean 更多 个人分类: Java https://blog ...

  7. 【线程池】ExecutorService与quartz搭配出现的问题

    问题描述: 使用quartz定时推送微信公众号模板消息,一分钟推送一次,定时器里面使用了一个ExecutorService线程池,大小为5个. 批量获取数据之后,全部数据都被分配到n/5的线程池里面等 ...

  8. python基本操作(四)

    与用户交互 为什么交互? 计算机取代人类,解放劳动力 如何交互 print('-'*100) input('请输入你的姓名:') print(""100) Python2和Pyth ...

  9. yield关键字有什么作用

    所属网站分类: python基础 > 语句 作者:goodbody 链接: http://www.pythonheidong.com/blog/article/10/ 来源:python黑洞网  ...

  10. JFinal 结合Dubbo发生的一些问题

    1.java.lang.NoSuchMethodError: org.jboss.resteasy.specimpl.BuiltResponse.getHeaders()Ljavax/ws/rs/co ...