题目很好,考察对主席树的深入理解与灵活运用。

首先看看一般解决中位数的思路,我们二分一个 \(mid\),将区间中 \(\ge mid\) 的数置为 \(1\),小于的置为 \(-1\),然后求区间和,若大于等于零则 \(mid\) 还能增大,否则减小。

现在就有了两个问题:第一,区间不固定;第二,每次二分一个答案就要重构区间,复杂度爆炸。

现在我们来仔细分析一下主席树的结构,首先,它是一个每个点都建了一棵线段树,形成前缀和的形式;每棵线段树又与区间有关。抽象地说,我们可以把第一个特征看作解决时间这一维限制,第二个特征解决位置这一维限制,即主席树同时解决了两维限制。

那再来找找这题的两维限制。如果我们把每次二分看做时间先后的操作,将每次二分的值作为一个“点”建线段树,就相当于预处理出了每次二分后区间的情况,省去了重构。再把权值离散化,那么就映射到了 \(1\sim n\) 的区间,对于 \(mid+1\),显然只有 \(mid\) 这个权值由 \(1\) 变成了 \(-1\) ,这其实只是一个单点修改的操作,这样就解决了第二个问题。

对于第一个问题,我们可以用最大子段和的思路维护 \(lmax,rmax,sum\) 的 tag,那么对于题目给出的区间 \([a,b],[c,d]\) ,答案即是 \([a,b]\) 的 \(rmax\)、\([b+1,c-1]\) 的 \(sum\),\([c,d]\) 的 \(lmax\) 之和。

哪里没有讲清楚可以看代码进一步理解。

#include <bits/stdc++.h>
#define l(x) t[x].l
#define r(x) t[x].r
using namespace std; const int N=1e5+5;
struct Tree
{
int l,r,lmax,rmax,sum;
void clear() {lmax=rmax=-N,l=r=sum=0;}
}t[N*20],Ans;
int n,Q,a[N],q[4],root[N],cntnode,id[N],ans; void build(int &rt,int l,int r)
{
t[rt=++cntnode]=(Tree){0,0,r-l+1,r-l+1,r-l+1};
if(l==r) return; int mid=l+r>>1;
build(l(rt),l,mid); build(r(rt),mid+1,r);
} inline void pushup(int rt)
{
t[rt].lmax=max(t[l(rt)].lmax,t[l(rt)].sum+t[r(rt)].lmax);
t[rt].rmax=max(t[r(rt)].rmax,t[r(rt)].sum+t[l(rt)].rmax);
t[rt].sum=t[l(rt)].sum+t[r(rt)].sum;
} void Insert(int &rt,int pre,int l,int r,int pos)
{
t[rt=++cntnode]=t[pre];
if(l==r) {t[rt].lmax=t[rt].rmax=t[rt].sum=-1; return;}
int mid=l+r>>1;
if(pos<=mid) Insert(l(rt),l(pre),l,mid,pos);
else Insert(r(rt),r(pre),mid+1,r,pos);
pushup(rt);
} void query(int rt,int lc,int rc,int l,int r)
{
if(l<=lc&&r>=rc)
{
Ans.lmax=max(Ans.lmax,Ans.sum+t[rt].lmax);
Ans.rmax=max(t[rt].rmax,Ans.rmax+t[rt].sum);
Ans.sum+=t[rt].sum;
return;
}
int mid=lc+rc>>1;
if(l<=mid) query(l(rt),lc,mid,l,r);
if(r>mid) query(r(rt),mid+1,rc,l,r);
} bool check(int mid)
{
int res=0;
if(q[1]+1<=q[2]-1)
Ans.clear(),query(root[mid],1,n,q[1]+1,q[2]-1),res+=Ans.sum;
Ans.clear(),query(root[mid],1,n,q[0],q[1]),res+=Ans.rmax;
Ans.clear(),query(root[mid],1,n,q[2],q[3]),res+=Ans.lmax;
return res>=0;
} int main()
{
scanf("%d",&n);
for(int i=1;i<=n;++i) scanf("%d",a+i),id[i]=i;
build(root[1],1,n);
sort(id+1,id+n+1,[](int x,int y){return a[x]<a[y];});
for(int i=2;i<=n;++i) Insert(root[i],root[i-1],1,n,id[i-1]);
scanf("%d",&Q);
while(Q--)
{
for(int i=0;i<4;++i)
scanf("%d",q+i),q[i]=(q[i]+ans)%n+1;
sort(q,q+4); int l=1,r=n;
while(l<=r)
{
int mid=l+r>>1;
if(check(mid)) ans=a[id[mid]],l=mid+1;
else r=mid-1;
}
printf("%d\n",ans);
}
return 0;
}

最新文章

  1. Struts2:Json插件_Ajax
  2. PHP 计算每个月的最后一天
  3. hbase批量数据导入报错:NotServingRegionException
  4. eval 与 Function
  5. C#中使用MySqlCommand执行插入语句后获取该数据主键id值的方法
  6. [转]epoll技术
  7. Ubuntu下将Sublime Text设置为默认编辑器
  8. 在企业级开发中使用Try...Catch...会影响效率吗?
  9. sqlserver 进行MD5加密
  10. 《30天自制操作系统》读书笔记(3) 引入C语言
  11. NuGet学习笔记(3)——搭建属于自己的NuGet服务器(转)
  12. 高质量程序设计指南C/C++语言——C++/C程序设计入门(3)
  13. SharePoint2013 Set Value To PeoplePicker
  14. June 30th. 2018, Week 26th. Saturday
  15. word文档最上面有一条不是页眉的线
  16. 袁创:使用反射动态调用ActiveX控件
  17. Json.Net(Newtonsoft)系列教程 4.Linq To JSON
  18. Android瀑布流优化,解决Recyclerview展示大批量图片时Item自动切换、闪烁、空白等问题
  19. Codeforces 1090A - Company Merging - [签到水题][2018-2019 Russia Open High School Programming Contest Problem A]
  20. 海西 &#183; 云交付 DevOps实践落地方案

热门文章

  1. Python_Selenium 之PO模式的思想、优化思路
  2. Git与GitHub入门
  3. 小白学k8s(7)helm[v3]使用了解
  4. redis cluster如何支持pipeline
  5. 合肥某小公司面试题:Spring基础
  6. 5、rsync全网备份
  7. SQL 小知识笔记
  8. Linux从头学02:x86中内存【段寻址】方式的来龙去脉
  9. Binding(五):多路绑定
  10. Mysql 主键的操作