bzoj 2653: middle (主席树+二分)
2653: middle
Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 2522  Solved: 1434
[Submit][Status][Discuss]
Description
Input
Output
Q行依次给出询问的答案。
Sample Input
170337785
271451044
22430280
969056313
206452321
3
3 1 0 2
2 3 1 4
3 1 4 0
Sample Output
271451044
969056313
HINT
Source
思路:
这道题比较容易想到二分答案,但是check答案比较难想
我们考虑check某个数的时候把大于它的值设为1,小于它的设为-1,那么对【b+1,c-1】区间取和,对【a,b】取右端最大,对【c,d】取左端最大加来如果大于等于0,那么说明值在右边,往右边二分否则往左边二分,
但是如果我们每次check重新建一遍线段树,那肯定是会超时的,我们用主席树存就好了,我们先将主席树上各个点的值赋为1,然后依次输入2-n,每次输入将前一个数赋值为-1(因为排完序后前一个数一定比当前数小),那么当 root [k] 时主席树上的值就等于之前对k建的线段树,主席树上我们维护三个值,一个是区间和,一个是右端最大,以及左端最大。
实现代码:
#include<bits/stdc++.h>
using namespace std;
#define mid int m = (l + r) >> 1
const int M = 2e4 + ;
int sum[M*],lsum[M*],rsum[M*],ls[M*],rs[M*],root[M],b[];
int idx,n;
struct node{
int val,id;
}a[M]; bool cmp(node x,node y){
return x.val < y.val;
} void pushup(int rt){
sum[rt] = sum[ls[rt]] + sum[rs[rt]];
lsum[rt] = max(lsum[ls[rt]],sum[ls[rt]]+lsum[rs[rt]]);
rsum[rt] = max(rsum[rs[rt]],sum[rs[rt]]+rsum[ls[rt]]);
} void build(int l,int r,int &rt){
rt = ++idx;
if(l == r){
sum[rt] = lsum[rt] = rsum[rt] = ;
return ;
}
mid;
build(l,m,ls[rt]); build(m+,r,rs[rt]);
pushup(rt);
} void update(int p,int l,int r,int old,int &rt){
rt = ++idx; ls[rt] = ls[old]; rs[rt] = rs[old];
if(l == r){
sum[rt] = lsum[rt] = rsum[rt] = -;
return ;
}
mid;
if(p <= m) update(p,l,m,ls[old],ls[rt]);
else update(p,m+,r,rs[old],rs[rt]);
pushup(rt);
} int query_sum(int L,int R,int l,int r,int rt){
if(L <= l&&R >= r){
return sum[rt];
}
mid;
int ret = ;
if(L <= m) ret += query_sum(L,R,l,m,ls[rt]);
if(R > m) ret += query_sum(L,R,m+,r,rs[rt]);
return ret;
} int query_lsum(int L,int R,int l,int r,int rt){
if(L <= l&&R >= r){
return lsum[rt];
}
mid;
if(R <= m) return query_lsum(L,R,l,m,ls[rt]);
else if(L > m) return query_lsum(L,R,m+,r,rs[rt]);
else return max(query_lsum(L,R,l,m,ls[rt]),query_sum(L,R,l,m,ls[rt])+query_lsum(L,R,m+,r,rs[rt]));
} int query_rsum(int L,int R,int l,int r,int rt){
if(L <= l&&R >= r){
return rsum[rt];
}
mid;
if(R <= m) return query_rsum(L,R,l,m,ls[rt]);
else if(L > m) return query_rsum(L,R,m+,r,rs[rt]);
else return max(query_rsum(L,R,m+,r,rs[rt]),query_sum(L,R,m+,r,rs[rt])+query_rsum(L,R,l,m,ls[rt]));
} bool check(int a,int b,int c,int d,int rt){
int cnt = ;
if(c- > b) cnt += query_sum(b+,c-,,n,root[rt]);
cnt += query_rsum(a,b,,n,root[rt]);
cnt += query_lsum(c,d,,n,root[rt]);
return cnt >= ;
} int main()
{
scanf("%d",&n);
for(int i = ;i <= n;i ++){
scanf("%d",&a[i].val);
a[i].id = i;
}
sort(a+,a++n,cmp);
build(,n,root[]);
for(int i = ;i <= n;i ++)
update(a[i-].id,,n,root[i-],root[i]);
int q,last = ;
scanf("%d",&q);
for(int i = ;i <= q;i ++){
scanf("%d%d%d%d",&b[],&b[],&b[],&b[]);
for(int j = ;j <= ;j ++)
b[j] = (b[j]+last)%n;
sort(b+,b+);
b[]++; b[]++; b[]++; b[]++;
int l = ,r = n,k;
while(l <= r){
mid;
if(check(b[],b[],b[],b[],m))
k = m,l = m+;
else r = m - ;
}
last = a[k].val;
printf("%d\n",last);
}
}
bzoj 2653: middle (主席树+二分)的更多相关文章
- BZOJ 2653: middle(主席树+二分答案)
		
传送门 解题思路 首先可以想到一种暴力做法,就是询问时二分,然后大于等于这个值的设为1,否则设为-1,然后就和GSS1那样统计答案.但是发现这样时间空间复杂度都很爆炸,所以考虑预处理,可以用主席树来做 ...
 - BZOJ 2653: middle 主席树 二分
		
https://www.lydsy.com/JudgeOnline/problem.php?id=2653 因为是两个方向向外延伸所以不能对编号取前缀和(这里只有前缀和向后传递的性质,不是实际意义的和 ...
 - bzoj 2653 middle(主席树)
		
题面:https://vjudge.net/problem/HYSBZ-2653 博客:https://blog.csdn.net/litble/article/details/78984846 这个 ...
 - BZOJ 2653: middle [主席树 中位数]
		
传送门 题意: 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你一个 长度为n的序列s.回答Q个这样的询问:s的左端点在[a,b]之间,右 ...
 - BZOJ 2653 middle | 主席树
		
题目: http://www.lydsy.com/JudgeOnline/problem.php?id=2653 题解: 设答案为ans,把大于等于ans的记为1,小于的记为-1,这样可以知道当前an ...
 - [BZOJ2653]middle 主席树+二分
		
2653: middle Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2042 Solved: 1123[Submit][Status][Disc ...
 - [BZOJ 2653] middle(可持久化线段树+二分答案)
		
[BZOJ 2653] middle(可持久化线段树+二分答案) 题面 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整. 给你一个长度为n的序 ...
 - BZOJ.1926.[SDOI2010]粟粟的书架(前缀和 主席树 二分)
		
题目链接 题意: 在给定矩形区域内找出最少的数,满足和>=k.输出数的个数.两种数据范围. 0~50 注意到(真没注意到...)P[i,j]<=1000,我们可以利用前缀和预处理. num ...
 - 2018湘潭邀请赛C题(主席树+二分)
		
题目地址:https://www.icpc.camp/contests/6CP5W4knRaIRgU 比赛的时候知道这题是用主席树+二分,可是当时没有学主席树,就连有模板都不敢套,因为代码实在是太长了 ...
 
随机推荐
- lower_bound函数与upper_bound函数
			
头文件 : algorithm vector<int>a a中的元素必须升序,用的是二分 lower_bound(a.begin(),a.end(),k) 返回a容器中,最右边的小于等于k ...
 - pandas删除某一列的方法
			
方法一:直接del df['column-name'] 删除sub_grade_列, 输入del df['sub_grade_x'] 方法二:采用drop方法,有下面三种等价的表达式: 1. df= ...
 - 如何优化Docker储存
			
大家在使用Docker的过程中,有没有想过,Docker在本地存储镜像时把文件存储在哪里了呢?有没有对文件的总大小做一定的限制呢?能不能调整本地存储的位置及总限制大小呢?今天,我们就从这些问题入手,来 ...
 - servlet总结:Servlet基础
			
Servlet基础 1.手工编写第一个Servlet ⑴继承HttpServlet ⑵重写doGet()或者doPost()方法 ⑶在web.xml中注册Servlet 2.使用eclipse编写第一 ...
 - git rebase的用法
			
改变基 一个git库,开发人员在master分支的Bcommit的时候,创建了一个dev分支,此时Bcommit是dev分支的基,然后分别进行两个分支的开发. 进行到master提交了Dcommit, ...
 - PHP导出CSV文件出现乱码的解决方法
			
在做项目时碰到使用外语的情况下,我们就会使用UTF-8编码.但是,在用PHP导出CSV文件时,如果写入的数据是使用UTF-8编码的日语.韩语之类的外文,就会出现乱码. 要解决PHP生成CSV文件的乱码 ...
 - 开发工具之Sublime编辑器
			
sublime是一款轻量级的编辑器,可以从官网上进行下载最新版本.它有很多使用并且强大的功能支持.例如:GOTO,package 等快捷操作.但有时候下载的版本不能进行安装package contro ...
 - 【Python3练习题 015】 一球从100米高度自由落下,每次落地后反跳回原高度的一半,再落下。求它在第10次落地时,共经过多少米?第10次反弹多高?
			
a = [100] #每个‘反弹落地’过程经过的路程,第1次只有落地(100米) h = 100 #每个‘反弹落地’过程,反弹的高度,第1次为100米 print('第1次从%s米高落地,走过%s ...
 - C\C++学习笔记 3
			
C++记录7 函数指针: 函数名为地址, 地址指的是在机器指令存储的地址. double func(int line){ reture line*3.5;} void f(int line, doub ...
 - [转帖]NUMA
			
作者:ibless 来源:CSDN 原文:https://blog.csdn.net/ibless/article/details/80114009 其实 很早之前对这一块有了解 比较多的的是 CCN ...