Luogu P2839 [国家集训队]middle
题目
首先我们考虑解决中位数一类问题的常用手段:二分\(mid\),将大于等于它的设为\(1\),小于它的设为\(−1\),判断区间和是否\(\ge0\)。
对于询问\(a,b,c,d\),二分完\(mid\)后,我们需要判断\([a,b]\)的最大后缀和\(+[c,d]\)的最大前缀和\(+(b,c)\)的和是否\(\ge0\)。
因为中位数一定是序列中出现过的数,所以我们可以排序后二分第\(mid\)大。
考虑使用主席树维护区间和区间最大后缀和区间最大前缀和。
最开始先把所有数设为\(1\)建树。
然后每次从小到大把序列中的一个树改为\(-1\)建一个新版本。
查询的时候按上面的来就可以了。
#include<bits/stdc++.h>
using namespace std;
namespace IO
{
char ibuf[(1<<21)+1],obuf[(1<<21)+1],st[20],*iS,*iT,*oS=obuf,*oT=obuf+(1<<21);
char Get(){return (iS==iT? (iT=(iS=ibuf)+fread(ibuf,1,(1<<21)+1,stdin),(iS==iT? EOF:*iS++)):*iS++);}
void Flush(){fwrite(obuf,1,oS-obuf,stdout),oS=obuf;}
void Put(char x){*oS++=x;if(oS==oT)Flush();}
int read(){int x=0;char ch=Get();while(ch>57||ch<48)ch=Get();while(ch>=48&&ch<=57)x=x*10+(ch^48),ch=Get();return x;}
void write(int x){int top=0;while(x)st[++top]=(x%10)+48,x/=10;while(top)Put(st[top--]);Put('\n');}
}
using namespace IO;
int max(int a,int b){return a>b? a:b;}
#define N 100007
#define mid ((l+r)>>1)
int root[N],q[4],id[N],a[N],ls[N<<4],rs[N<<4],cnt,n;
struct node{int lv,rv,sum;void init(){lv=rv=-N,sum=0;}}Ans,v[N<<4];
int cmp(int x,int y){return a[x]<a[y];}
node merge(node a,node b)
{
node c;
c.lv=max(a.lv,a.sum+b.lv);
c.rv=max(b.rv,b.sum+a.rv);
c.sum=a.sum+b.sum;
return c;
}
void build(int &p,int l,int r)
{
p=++cnt,v[p].lv=v[p].rv=v[p].sum=r-l+1;
if(l^r) build(ls[p],l,mid),build(rs[p],mid+1,r);
}
void modify(int &p,int l,int r,int x)
{
v[++cnt]=v[p],ls[cnt]=ls[p],rs[cnt]=rs[p],p=cnt;
if(l==r) return (void)(v[p].lv=v[p].rv=v[p].sum=-1);
(x<=mid? modify(ls[p],l,mid,x):modify(rs[p],mid+1,r,x)),v[p]=merge(v[ls[p]],v[rs[p]]);
}
void query(int p,int l,int r,int L,int R)
{
if(L<=l&&r<=R) return (void)(Ans=merge(Ans,v[p]));
if(L<=mid) query(ls[p],l,mid,L,R);
if(R>mid) query(rs[p],mid+1,r,L,R);
}
int check(int p)
{
int sum=0;
if(q[1]+1<=q[2]-1) Ans.init(),query(root[p],1,n,q[1]+1,q[2]-1),sum+=Ans.sum;
Ans.init(),query(root[p],1,n,q[0],q[1]),sum+=Ans.rv;
Ans.init(),query(root[p],1,n,q[2],q[3]),sum+=Ans.lv;
return sum>=0;
}
int main()
{
int i,Q,ans,l,r;
n=read(),build(root[1],1,n),v[0].init();
for(i=1;i<=n;++i) a[i]=read(),id[i]=i;
sort(id+1,id+n+1,cmp);
for(i=2;i<=n;++i) root[i]=root[i-1],modify(root[i],1,n,id[i-1]);
for(Q=read(),ans=0;Q;--Q)
{
for(i=0;i<4;++i) q[i]=(read()+ans)%n+1;
sort(q,q+4),l=1,r=n;
while(l<=r) check(mid)? (ans=a[id[mid]],l=mid+1):(r=mid-1);
write(ans);
}
return Flush(),0;
}
Luogu P2839 [国家集训队]middle的更多相关文章
- P2839 [国家集训队]middle
P2839 [国家集训队]middle 好妙的题啊,,,, 首先二分一个答案k,把数列里>=k的数置为1,=0就是k>=中位数,<0就是k<中位数 数列的最大和很好求哇 左边的 ...
- [洛谷P2839][国家集训队]middle
题目大意:给你一个长度为$n$的序列$s$.$Q$个询问,问在$s$中的左端点在$[a,b]$之间,右端点在$[c,d]$之间的子段中,最大的中位数. 强制在线. 题解:区间中位数?二分答案,如果询问 ...
- Luogu 2839 [国家集训队]middle
感觉这题挺好的. 首先对于中位数最大有一个很经典的处理方法就是二分,每次二分一个数组中的下标$mid$,然后我们把$mid$代回到原来的数组中检查,如果一个数$a_{i} \geq mid$,那么就把 ...
- 洛谷P2839 [国家集训队]middle 主席树_二分
Code: #include <cstdio> #include <algorithm> #include <cstring> #include <strin ...
- [国家集训队]middle 解题报告
[国家集训队]middle 主席树的想法感觉挺妙的,但是这题数据范围这么小,直接分块草过去不就好了吗 二分是要二分的,把\(<x\)置\(-1\),\(\ge x\)的置\(1\),于是我们需要 ...
- [国家集训队]middle
[国家集训队]middle 题目 解法 开\(n\)颗线段树,将第\(i\)颗线段树中大于等于第\(i\)小的数权值赋为1,其他的则为-1,对于每个区间维护一个区间和,最大前缀和,最大后缀和. 然后二 ...
- luogu P2757 [国家集训队]等差子序列
题目链接 luogu P2757 [国家集训队]等差子序列 题解 线段树好题 我选择暴力 代码 // luogu-judger-enable-o2 #include<cstdio> inl ...
- luogu P2619 [国家集训队2]Tree I
题目链接 luogu P2619 [国家集训队2]Tree I 题解 普通思路就不说了二分增量,生成树check 说一下坑点 二分时,若黑白边权有相同,因为权值相同优先选白边,若在最有增量时出现黑白等 ...
- CF484E Sign on Fence && [国家集训队]middle
CF484E Sign on Fence #include<bits/stdc++.h> #define RG register #define IL inline #define _ 1 ...
随机推荐
- jquery的checked
目前使用的jQuery版本为 v1.11.2 jquery判断checked的三种方法: .attr('checked'): //看版本1.6+返回:"checked"或&qu ...
- C# 选择文件夹 选择文件
选择文件 //选择文件 OpenFileDialog dialog = new OpenFileDialog(); dialog.Multiselect = true;//该值确定是否可以选择多个文件 ...
- FFT用于高效大数乘法(当模板用)
转载来源:https://blog.csdn.net/zj_whu/article/details/72954766 #include <cstdio> #include <cmat ...
- [BZOJ5249][九省联考2018]IIIDX:线段树+贪心
分析 GXZlegend orz 构造出一组合法的解并不是难事,但是我们需要输出的是字典序最大的解. 字典序最大有另一种理解方式,就是让越小的数尽量越靠后. 我们从树的根结点出发,从1开始填数,构造出 ...
- 大哥带的XSS练习LEVE3
0X01DOM-XSS进阶之inner显式输出 首先我们先了解一下DOM型和和其他到底有什么区别 dom就是一个树状的模型,你可以编写Javascript代码根据dom一层一层的节点,去遍历/获取/修 ...
- [CSP-S模拟测试]:打表(猜测题意+结论)
题目传送门(内部题139) 输入格式 第一行两个整数$k,ans$,表示内存地址$A$的位数,以及答案所在的内存地址. 接下来一行$2^k$个整数,分别表示内存地址$0...2^k-1$上的值. 输出 ...
- 分布式-信息方式-JMS Queue示例
代码 package test.mq.helloword; import javax.jms.Connection; import javax.jms.ConnectionFactory; impor ...
- LeetCode 98. 验证二叉搜索树(Validate Binary Search Tree)
题目描述 给定一个二叉树,判断其是否是一个有效的二叉搜索树. 假设一个二叉搜索树具有如下特征: 节点的左子树只包含小于当前节点的数. 节点的右子树只包含大于当前节点的数. 所有左子树和右子树自身必须也 ...
- 如何消除 com.netflix.discovery.shared.transport.TransportException: Cannot execute request on any known server
在application.properties中添加以下两句话: eureka.client.register-with-eureka=falseeureka.client.fetch-registr ...
- win10 点击开始按钮无反应
本人亲身经历 由于安装软件时需要注册表权限,在一顿猛如虎的操作下,将注册表中 HKEY_CURRENT_USER 的权限出问题.而导致无法打开 开始菜单 ----------------以下是本人为了 ...