Luogu2839 [国家集训队]middle 题解
题目很好,考察对主席树的深入理解与灵活运用。
首先看看一般解决中位数的思路,我们二分一个 \(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;
}
Luogu2839 [国家集训队]middle 题解的更多相关文章
- luogu2839 [国家集训队]middle
题目链接:洛谷 题目大意:给定一个长度为$n$的序列,每次询问左端点在$[a,b]$,右端点在$[c,d]$的所有子区间的中位数的最大值.(强制在线) 这里的中位数定义为,对于一个长度为$n$的序列排 ...
- 【LG2839】[国家集训队]middle
[LG2839][国家集训队]middle 题面 洛谷 题解 按照求中位数的套路,我们二分答案\(mid\),将大于等于\(mid\)的数设为\(1\),否则为\(-1\). 若一个区间和大于等于\( ...
- [国家集训队]middle 解题报告
[国家集训队]middle 主席树的想法感觉挺妙的,但是这题数据范围这么小,直接分块草过去不就好了吗 二分是要二分的,把\(<x\)置\(-1\),\(\ge x\)的置\(1\),于是我们需要 ...
- [国家集训队]middle
[国家集训队]middle 题目 解法 开\(n\)颗线段树,将第\(i\)颗线段树中大于等于第\(i\)小的数权值赋为1,其他的则为-1,对于每个区间维护一个区间和,最大前缀和,最大后缀和. 然后二 ...
- P2839 [国家集训队]middle
P2839 [国家集训队]middle 好妙的题啊,,,, 首先二分一个答案k,把数列里>=k的数置为1,=0就是k>=中位数,<0就是k<中位数 数列的最大和很好求哇 左边的 ...
- CF484E Sign on Fence && [国家集训队]middle
CF484E Sign on Fence #include<bits/stdc++.h> #define RG register #define IL inline #define _ 1 ...
- [洛谷P2839][国家集训队]middle
题目大意:给你一个长度为$n$的序列$s$.$Q$个询问,问在$s$中的左端点在$[a,b]$之间,右端点在$[c,d]$之间的子段中,最大的中位数. 强制在线. 题解:区间中位数?二分答案,如果询问 ...
- BZOJ.2653.[国家集训队]middle(可持久化线段树 二分)
BZOJ 洛谷 求中位数除了\(sort\)还有什么方法?二分一个数\(x\),把\(<x\)的数全设成\(-1\),\(\geq x\)的数设成\(1\),判断序列和是否非负. 对于询问\(( ...
- 解题:国家集训队 Middle
题面 求中位数的套路:二分,大于等于的设为1,小于的设为-1 于是可以从小到大排序后依次加入可持久化线段树,这样每次只会变化一个位置 那左右端点是区间怎么办? 先把中间的算上,然后维护每个区间左右两侧 ...
随机推荐
- 【NX二次开发】开发好几年,还只会用ufusr?其他用户出口函数介绍
用户出口(User Exit)是NX Open 中的一个重要概念.NX在运行过程中某些特定的位置存在规定的出口,当进程执行到这些出口时,NX会自动检查用户是否在此处已定义了指向内部程序位置的环境变量: ...
- DDD实战课(实战篇)--学习笔记
目录 DDD实践:如何用DDD重构中台业务模型? 领域建模:如何用事件风暴构建领域模型? 代码模型(上):如何使用DDD设计微服务代码模型? 代码模型(下):如何保证领域模型与代码模型的一致性? 边界 ...
- 白日梦的MySQL专题(第38篇文章)8分钟回顾MySQL的索引
目录 公众号首发-推荐阅读原文-格式更好看 一.导读 二.聚簇索引 三.二级索引 四.联合索引 4.1.什么是联合索引 4.2.左前缀原则 4.3.联合索引的分组&排序 五.覆盖索引 六.倒排 ...
- 约会Rendezvous
约会 Rendezvous 内存限制:128 MiB 时间限制:1000 ms 标准输入输出 题目描述 给定一个有 nnn 个顶点的有向图,每个顶点有且仅有一条出边.每次询问给出两个顶点 ai ...
- Mysql优化(出自官方文档) - 第七篇
Mysql优化(出自官方文档) - 第七篇 目录 Mysql优化(出自官方文档) - 第七篇 Optimizing Data Change Statements 1 Optimizing INSERT ...
- SpringBoot实战:10分钟快速搞定环境
什么是 springboot Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程. 该框架使用了特定的方式来进行配置,从而使开发人 ...
- 高性能的Redis之对象底层实现原理详解
对象 在前面的数个章节里, 我们陆续介绍了 Redis 用到的所有主要数据结构, 比如简单动态字符串(SDS).双端链表.字典.压缩列表.整数集合, 等等. Redis 并没有直接使用这些数据结构来实 ...
- 温故知新Docker概念及Docker Desktop For Windows v3.1.0安装
Docker 简介 什么是Docker? Docker是一个开放源代码软件项目,项目主要代码在2013年开源于GitHub.它是云服务技术上的一次创新,让应用程序布署在软件容器下的工作可以自动化进行, ...
- .net core 支付宝,微信支付 一
源码: https://github.com/aspros-luo/Qwerty.Payment/tree/develop 支付宝支付:参考支付宝sdk及文档,https://docs.open.al ...
- 04 jumpserver资产管理
4.资产管理: (1)管理用户: 管理用户是资产(被控服务器)上的 root,或拥有 NOPASSWD: ALL sudo 权限的用户, JumpServer 使用该用户来 `推送系统用户`.`获取资 ...