【bzoj2653】【middle】【主席树+二分答案】
Description
一个长度为 n 的序列 a ,设其排过序之后为 b ,其中位数定义为 b[n/2] ,其中 a,b 从 0 开始标号 , 除法取下整。
给你一个长度为 n 的序列 s 。回答 Q 个这样的询问 : s 的左端点在 [a,b] 之间 , 右端点在 [c,d] 之间的子序列中 ,最大的中位数。
其中 a
Solution
对着题解理解了半天……又对着代码调了半天……最后发现竟然是一个函数名没写orz
不过不得不说这题真的是主席树好题
先考虑二分答案,找出区间内比mid小的数有多少
因为对答案的贡献只有有或没有,所以可以把比mid小的都设为-1,比mid大的都设为1,如果区间内的和大于等于0,说明mid可行,继续二分下去
然而如果离散之后对每一个值建树,空间毫无疑问爆炸
于是只要用主席树维护一下就可以了
//minamoto
#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
#define getc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
char buf[<<],*p1=buf,*p2=buf;
inline int read(){
#define num ch-'0'
char ch;bool flag=;int res;
while(!isdigit(ch=getc()))
(ch=='-')&&(flag=true);
for(res=num;isdigit(ch=getc());res=res*+num);
(flag)&&(res=-res);
#undef num
return res;
}
char obuf[<<],*o=obuf;
inline void print(int x){
if(x>) print(x/);
*o++=x%+;
}
const int N=,M=N*;
int n,Pre,q,cnt;
int rt[N],p[];
struct node{
int l,r,lmx,rmx,sum;
}t[M],op;
struct data{
int x,id;
inline bool operator <(const data &b)const
{return x<b.x;}
}a[N];
inline void pushup(int x){
t[x].sum=t[t[x].l].sum+t[t[x].r].sum;
t[x].lmx=max(t[t[x].l].lmx,t[t[x].l].sum+t[t[x].r].lmx);
t[x].rmx=max(t[t[x].r].rmx,t[t[x].r].sum+t[t[x].l].rmx);
}
void build(int &now,int l,int r){
now=++cnt;
if(l==r){t[now].lmx=t[now].rmx=t[now].sum=;return;}
int mid=(l+r)>>;
build(t[now].l,l,mid);
build(t[now].r,mid+,r);
pushup(now);
}
void update(int last,int &now,int l,int r,int k){
now=++cnt;
if(l==r){t[now].lmx=t[now].rmx=t[now].sum=-;return;}
int mid=(l+r)>>;
if(k<=mid) t[now].r=t[last].r,update(t[last].l,t[now].l,l,mid,k);
else t[now].l=t[last].l,update(t[last].r,t[now].r,mid+,r,k);
pushup(now);
}
node merge(node x,node y){
node z;
z.sum=x.sum+y.sum;
z.lmx=max(x.lmx,x.sum+y.lmx);
z.rmx=max(y.rmx,y.sum+x.rmx);
return z;
}
node find(int x,int l,int r,int y,int z){
if(y>z) return op;
if(l==y&&r==z) return t[x];
int mid=(l+r)>>;
if(z<=mid) return find(t[x].l,l,mid,y,z);
else if(y>mid) return find(t[x].r,mid+,r,y,z);
else return merge(find(t[x].l,l,mid,y,mid),find(t[x].r,mid+,r,mid+,z));
}
int query(int x){
return find(rt[x],,n,p[],p[]).rmx+find(rt[x],,n,p[]+,p[]-).sum+find(rt[x],,n,p[],p[]).lmx;
}
int main(){
//freopen("testdata.in","r",stdin);
n=read();
for(int i=;i<=n;++i) a[i].x=read(),a[i].id=i;
sort(a+,a++n);
build(rt[],,n);
for(int i=;i<=n;++i) update(rt[i-],rt[i],,n,a[i-].id);
q=read();
while(q--){
int x=read(),y=read(),z=read(),k=read();
p[]=(x+Pre)%n+,p[]=(y+Pre)%n+,p[]=(z+Pre)%n+,p[]=(k+Pre)%n+;
sort(p+,p+);
int l=,r=n,ans=;
while(l<=r){
int mid=(l+r)>>;
int f=query(mid);
if(f>=) ans=mid,l=mid+;
else r=mid-;
}
Pre=a[ans].x;
print(a[ans].x),*o++='\n';
}
fwrite(obuf,o-obuf,,stdout);
return ;
}
【bzoj2653】【middle】【主席树+二分答案】的更多相关文章
- [BZOJ2653]middle 主席树+二分
2653: middle Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2042 Solved: 1123[Submit][Status][Disc ...
- BZOJ 2653: middle(主席树+二分答案)
传送门 解题思路 首先可以想到一种暴力做法,就是询问时二分,然后大于等于这个值的设为1,否则设为-1,然后就和GSS1那样统计答案.但是发现这样时间空间复杂度都很爆炸,所以考虑预处理,可以用主席树来做 ...
- bzoj 2653: middle (主席树+二分)
2653: middle Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2522 Solved: 1434[Submit][Status][Disc ...
- BZOJ 1926: [Sdoi2010]粟粟的书架(主席树,二分答案)
BZOJ 1926: [Sdoi2010]粟粟的书架(主席树,二分答案) 题意 : 给你一个长为\(R\)宽为\(C\)的矩阵,第\(i\)行\(j\)列的数为\(P_{i,j}\). 有\(m\)次 ...
- BZOJ5343[Ctsc2018]混合果汁——主席树+二分答案
题目链接: CTSC2018混合果汁 显然如果美味度高的合法那么美味度低的一定合法,因为美味度低的可选方案包含美味度高的可选方案. 那么我们二分一个美味度作为答案然后考虑如何验证? 选择时显然要贪心的 ...
- P4094 [HEOI2016/TJOI2016]字符串 后缀数组+主席树+二分答案
$ \color{#0066ff}{ 题目描述 }$ 佳媛姐姐过生日的时候,她的小伙伴从某东上买了一个生日礼物.生日礼物放在一个神奇的箱子中.箱子外边写了一个长为n的字符串s,和m个问题.佳媛姐姐必须 ...
- BZOJ 4556 [Tjoi2016&Heoi2016]字符串 ——后缀数组 ST表 主席树 二分答案
Solution 1: 后缀数组暴力大法好 #include <map> #include <cmath> #include <queue> #include &l ...
- HDU - 6621 K-th Closest Distance 主席树+二分答案
K-th Closest Distance 主席树第二波~ 题意 给你\(n\)个数\(m\)个询问,问\(i\in [l,r]\)计算每一个\(|a_{i}-p|\)求出第\(k\)小 题目要求强制 ...
- BZOJ 2653: middle 主席树 二分
https://www.lydsy.com/JudgeOnline/problem.php?id=2653 因为是两个方向向外延伸所以不能对编号取前缀和(这里只有前缀和向后传递的性质,不是实际意义的和 ...
随机推荐
- bzoj1007/luogu3194 水平可见直线 (单调栈)
先按斜率从小到大排序,然后如果排在后面的点B和前面的点A的交点是P,那B会把A在P的右半段覆盖掉,A会把B在P的左半段覆盖掉. 然后如果我们现在又进来了一条线,它跟上一条的交点还在上一条和上上条的左边 ...
- 利用NEST2.0 在C#中操作Elasticsearch
前言:本文主要演示了如何通过c#来操作elasticsearch,分两个方面来演示: 索引数据 搜索数据 Note: 注意我索引数据和搜索数据是两个不同的例子,没有前后依赖关系 准备工作:需要在vis ...
- A1054. The Dominant Color
Behind the scenes in the computer's memory, color is always talked about as a series of 24 bits of i ...
- codeblocks 支持多个exe同时执行
如果看总时间,没什么用,因为总资源是一样的. 但是可以做到:吃饭前,执行多个程序,吃完饭,所有程序执行完.
- NCBI上查看SNP位点在哪个基因座上(locus)
首先,进入NCBI的主页网站:https://www.ncbi.nlm.nih.gov/variation/view/ 进入后,在下图红色框框位置输入目的SNP,比如rs608139 输完后,出现如下 ...
- Visio画UML类图、序列图 for Java
参考文档: 1.百度搜索: 怎样用Visio 2007画C++类图 连接 https://jingyan.baidu.com/article/9f7e7ec07286e16f281554f7.html ...
- NO.11 复制时勿忘其每个成分
1.Coping 函数应该确保复制对象内的"每一个成员变量",和调用合适的 "base class"构造函数(base class 某些成员往往是private ...
- 邮件发送-》http://service.mail.qq.com/cgi-bin/help?subtype=1&&id=28&&no=1001256
昨天使用发送邮件报了一个错 http://service.mail.qq.com/cgi-bin/help?subtype=1&&id=28&&no=1001256 主 ...
- 【codeforces】【Round#523D】TV shows
题意:n个节目,每个节目的播放时间为[li,ri],你需要选择一些电视机全部播放这些节目,一台电视机不能同时播放多个节目,选择一个新的电视机代价为x , 如果某台电视机的使用时间为[Li,Ri]需要付 ...
- 线程的同步(协调)synchronized
[格式] 同步代码块:synchronized(Object){...} 关键字在代码块前,每次只允许一个线程调用此代码块. Object为任何对象(一般用this),每个对象都有一个标志位(0锁住状 ...