BZOJ 2653 middle
AC通道:http://www.lydsy.com/JudgeOnline/problem.php?id=2653
题目大意:多组询问,求左右端点在规定范围内移动所能得到的最大中位数。
[分析]
求中位数这类题目比较少见[笔者见识少...],但是它的性质比较优美——从小到大排序排在中间的数字。
如果往常求一个无序数列的中位数,最好的复杂度就是n*logn[快排一遍的复杂度],[因为你要知道每个元素在序列中的大小]
但是我们这里是多组询问,而且元素不会修改,也就是每个元素的相对大小会在一开始预处理就搞定。
现在我们再来想:左右端点只是一个范围,怎么求出满足情况的最优区间呢?
根据时间的效率,肯定不能用枚举。
那么这个区间就一定有性质,什么性质呢?中位数最大。
我们要从中位数最大来找到这个区间。
也就是这个区间里比中位数大的元素和比中位数小的元素一样多。
可想而知:我们希望这个区间内的大的数越多,小的数越少越好,可是大小只是一个相对的概念,必须要和中位数比较才知道什么叫大,什么叫小。
谁是最大的中位数呢?求的就是这个,我们怎么知道...那我们只好二分了。
二分出这个中位数之后,我们要找这个最优区间就好办了:
假设比中位数小的数值记作-1,比中位数大的数值记作1。
最优区间一定满足:区间元素[指的是1,-1这些数]的和最大。
求最大和的这个过程可以用很多数据结构解决了,即求[a,b]的右端最大+(b,c)的和+[c,d]的左端最大。
可是我其实只需要这个值大于0,当前二分的值就是合法的了,我就会去追求一个更大的中位数,对吧。
也就是说,思考的过程是:
二分一个答案->找到最优的区间->区间如果满足->去追寻更好的答案。
其中最优区间是找寻答案的前提,但只有先暂时确定一个较好值才能找到符合这个的最优区间。[逻辑上有点成环]
最后因为对于每个二分出来的中位数值都希望有一个[-1,1]的序列专门分给它,这个靠什么呢?
想起之前的可持久化线段树就好办了:
树的节点意义是下标的位置范围,树的节点同时需要记录下这个节点代表区间的“右端最大”、“左端最大”、“求和”三种属性
最小的元素所有位置都标上1[因为所有元素都比它大嘛...],
然后从小到大的添加元素,先复制上一棵树,每次将树上上一个元素所在的位置标上-1,同时需要在过程中更新出右端最大、左端最大、求和三种属性。
因为上一棵树和这一棵树的唯一区别就是上一棵树在上一个元素的位置上标记为1,而这一棵树标记为-1,这样每次就是更新树上的一条链,变化不会很大,采用的就是可持久化线段树的思想了。
最后献上代码:
#include<cstdio>
#include<cstring>
#include<algorithm> using namespace std; inline int in(){
int x=;char ch=getchar();
while(ch>'' || ch<'') ch=getchar();
while(ch>='' && ch<='') x=x*+ch-'',ch=getchar();
return x;
} const int maxn=; int n,m,key,cnt,ans;
int a[maxn];
int id[maxn],rt[maxn]; struct Node{
int l,r,sz;
int lx,rx,sm;
}s[maxn*]; bool cmp(const int &x1,const int &x2){
return a[x1]<a[x2];
} void renew(int x){
s[x].sm=s[s[x].l].sm+s[s[x].r].sm;
s[x].lx=max(s[s[x].l].lx,s[s[x].l].sm+s[s[x].r].lx);
s[x].rx=max(s[s[x].r].rx,s[s[x].r].sm+s[s[x].l].rx);
} void build(int l,int r,int &rt){
if(l==r){
rt=++cnt,s[rt].sm=s[rt].lx=s[rt].rx=;
return ;
}
rt=++cnt;
int mid=(l+r)>>;
build(l,mid,s[rt].l);
build(mid+,r,s[rt].r);
renew(rt);
} void update(int last,int l,int r,int &rt,int val){
rt=++cnt;s[rt]=s[last];
if(l==r){
s[rt].lx=s[rt].rx=s[rt].sm=val;
return ;
}
int mid=(l+r)>>;
if(key<=mid) update(s[last].l,l,mid,s[rt].l,val);
else update(s[last].r,mid+,r,s[rt].r,val);
renew(rt);
} int get_all(int rt,int l,int r,int x,int y){
if(l==x && r==y) return s[rt].sm;
int mid=(l+r)>>;
if(y<=mid)
return get_all(s[rt].l,l,mid,x,y);
else if(x>mid)
return get_all(s[rt].r,mid+,r,x,y);
else
return get_all(s[rt].l,l,mid,x,mid)+get_all(s[rt].r,mid+,r,mid+,y);
} int get_lx(int rt,int l,int r,int x,int y){
if(l==x && r==y) return s[rt].lx;
int mid=(l+r)>>;
if(y<=mid)
return get_lx(s[rt].l,l,mid,x,y);
else if(x>mid)
return get_lx(s[rt].r,mid+,r,x,y);
else
return max(get_lx(s[rt].l,l,mid,x,mid),get_all(s[rt].l,l,mid,x,mid)+get_lx(s[rt].r,mid+,r,mid+,y));
} int get_rx(int rt,int l,int r,int x,int y){
if(l==x && r==y) return s[rt].rx;
int mid=(l+r)>>;
if(y<=mid)
return get_rx(s[rt].l,l,mid,x,y);
else if(x>mid)
return get_rx(s[rt].r,mid+,r,x,y);
else
return max(get_rx(s[rt].r,mid+,r,mid+,y),get_all(s[rt].r,mid+,r,mid+,y)+get_rx(s[rt].l,l,mid,x,mid));
} bool check(int k,int a,int b,int c,int d){
int sum=;
if(c>b+) sum+=get_all(rt[k],,n-,b+,c-);
sum+=get_rx(rt[k],,n-,a,b);
sum+=get_lx(rt[k],,n-,c,d);
return sum>=;
} int main(){
#ifndef ONLINE_JUDGE
freopen("2653.in","r",stdin);
freopen("2653.out","w",stdout);
#endif n=in();
for(int i=;i<n;i++)
a[i]=in(),id[i]=i;
sort(id,id+n,cmp);
build(,n-,rt[]);
for(int i=;i<n;i++)
key=id[i-],update(rt[i-],,n-,rt[i],-); int ord[]; m=in();
while(m--){
ord[]=in(),ord[]=in(),ord[]=in(),ord[]=in();
for(int i=;i<;i++)
ord[i]=(ord[i]+ans)%n;
sort(ord,ord+);
int l=,r=n,mid;
while(l+<r){
mid=(l+r)>>;
if(check(mid,ord[],ord[],ord[],ord[])) l=mid;
else r=mid;
}
ans=a[id[l]];
printf("%d\n",ans);
} return ;
}
BZOJ 2653 middle的更多相关文章
- [BZOJ 2653] middle(可持久化线段树+二分答案)
[BZOJ 2653] middle(可持久化线段树+二分答案) 题面 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整. 给你一个长度为n的序 ...
- bzoj 2653: middle (主席树+二分)
2653: middle Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2522 Solved: 1434[Submit][Status][Disc ...
- BZOJ 2653: middle 主席树 二分
https://www.lydsy.com/JudgeOnline/problem.php?id=2653 因为是两个方向向外延伸所以不能对编号取前缀和(这里只有前缀和向后传递的性质,不是实际意义的和 ...
- BZOJ 2653 middle | 主席树
题目: http://www.lydsy.com/JudgeOnline/problem.php?id=2653 题解: 设答案为ans,把大于等于ans的记为1,小于的记为-1,这样可以知道当前an ...
- bzoj 2653 middle (可持久化线段树)
middle Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1981 Solved: 1097[Submit][Status][Discuss] D ...
- BZOJ 2653: middle [主席树 中位数]
传送门 题意: 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你一个 长度为n的序列s.回答Q个这样的询问:s的左端点在[a,b]之间,右 ...
- bzoj 2653 middle 二分答案 主席树判定
判断中位数是否可行需要将当前的解作为分界,大于其的置为1,小于为-1,然后b-c必选,ab,cd可不选,这个用线段树判定就好 但不能每次跑,所以套主席树,按权值排序,构建主席树,更新时将上一个节点改为 ...
- BZOJ 2653 middle 二分答案+可持久化线段树
题目大意:有一个序列,包含多次询问.询问区间左右端点在规定区间里移动所得到的最大中位数的值. 考虑对于每个询问,如何得到最优区间?枚举显然是超时的,只能考虑二分. 中位数的定义是在一个序列中,比中位数 ...
- bzoj 2653 middle(主席树)
题面:https://vjudge.net/problem/HYSBZ-2653 博客:https://blog.csdn.net/litble/article/details/78984846 这个 ...
随机推荐
- 深入浅出MongoDB(一)NoSQL
从本文开始,我们一起学习一下MongoDB相关内容,在学习MongoDB之前,首先要做的就是学习NoSQL. 为什么要学习NoSQL,原因很简单,因为MongoDB是NoSQL数据库的一种,换言之,如 ...
- PHP,Mysql-根据一个给定经纬度的点,进行附近地点查询–合理利用算法,效率提高2125倍
目前的工作是需要对用户的一些数据进行分析,每个用户都有若干条记录,每条记录中有用户的一个位置,是用经度和纬度表示的.还有一个给定的数据库,存储的是一些已知地点以及他们的经纬度,内有43W多条的数据.现 ...
- 生产WCF客户端类文件的命令格式
生产WCF客户端类文件的命令格式: svcutil.exe net.tcp://127.0.0.1:8732/ChromaMI.Remote.ConfigService/RemoteConfigSer ...
- xdebug
必须的有4个(remote_enable 默认没开,所以必要),其它默认值一般可以xdebug.remote_enable=On; //这个是必须xdebug.remote_host=192.168. ...
- WPF.UIShell UIFramework之自定义窗口的深度技术 - 模态闪动(Blink)、窗口四边拖拽支持(WmNCHitTest)、自定义最大化位置和大小(WmGetMinMaxInfo)
无论是在工作和学习中使用WPF时,我们通常都会接触到CustomControl,今天我们就CustomWindow之后的一些边角技术进行探讨和剖析. 窗口(对话框)模态闪动(Blink) 自定义窗口的 ...
- DrawTool多重笔之前奏 => 通过InkAnalyzer实现图形识别
这里要介绍的是通过InkAnalyzer来实现简单图形的识别,例如圆,椭圆,正方形,三角形等,当然你也可以通过扩展来实现自定义图形的识别,在使用InkAnalyzer前,你需要引用IAWinFX.dl ...
- hdu 4631Sad Love Story<计算几何>
链接:http://acm.hdu.edu.cn/showproblem.php?pid=4631 题意:依次给你n个点,每次求出当前点中的最近点对,输出所有最近点对的和: 思路:按照x排序,然后用s ...
- hdu 1874 畅通工程续
题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=1874 畅通工程续 Description 某省自从实行了很多年的畅通工程计划后,终于修建了很多路.不过 ...
- 7.Knockout.Js(Mapping插件)
前言 Knockout设计成允许你使用任何JavaScript对象作为view model.必须view model的一些属性是observable的,你可以使用KO绑定他们到你的UI元素上,当这些o ...
- 为iPhone6设计自适应布局
Apple从iOS6加入了Auto Layout后开始就比较委婉的开始鼓励.建议开发者使用自适应布局,但是到目前为止,我感觉大多数开发者一直在回避这个问题,不管是不是由于历史原因造成的,至少他们在心底 ...