2653: middle

链接

分析:

  二分答案+主席树。

  对于中位数的经典做法,就是二分一个数,将小于的变成-1,大于等于的变成+1,那么如果sum>=0(因为+1包括等于),L=mid+1,否则R=mid-1。

  那么考虑二分一个中位数(当然只二分出现过的数即可),然后向上面一样判断。

  因为二分的数字只有n个,可以建立n颗只包含-1和+1的权值线段树,发现第i小的权值线段树与第i+1小的权值线段树只有一个位置不同,所以可以类似可持久化的思路,每次只去建立一条链。

  查询的区间有很多个,不能挨个查询它们的和,由于查询最大值,所以可以再在线段树上维护左端最大子段和,右端最大子段和。

代码:

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<cmath>
#include<cctype>
#include<set>
#include<queue>
#include<vector>
#include<map>
#define pa pair<int,int>
using namespace std;
typedef long long LL; inline int read() {
int x=,f=;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-;
for(;isdigit(ch);ch=getchar())x=x*+ch-'';return x*f;
} const int N = ;
struct Node{
int sum, Lmx, Rmx;
}T[N * ];
int ls[N * ], rs[N * ], Root[N], Index, n;
pa A[N]; Node operator + (const Node &A,const Node &B) {
Node res;
res.sum = A.sum + B.sum;
res.Lmx = max(A.Lmx, A.sum + B.Lmx);
res.Rmx = max(B.Rmx, A.Rmx + B.sum);
return res;
}
void build(int l,int r,int &rt) {
rt = ++Index;
if (l == r) {
T[rt].sum = T[rt].Lmx = T[rt].Rmx = ; return ;
}
int mid = (l + r) >> ;
build(l, mid, ls[rt]); build(mid + , r, rs[rt]);
T[rt] = T[ls[rt]] + T[rs[rt]];
}
void update(int l,int r,int &rt,int pre,int p) {
if (!rt) rt = ++Index;
if (l == r) {
T[rt].sum = T[rt].Lmx = T[rt].Rmx = -; return ;
}
int mid = (l + r) >> ;
if (p <= mid) {
rs[rt] = rs[pre];
update(l, mid, ls[rt], ls[pre], p);
} else {
ls[rt] = ls[pre];
update(mid + , r, rs[rt], rs[pre], p);
}
T[rt] = T[ls[rt]] + T[rs[rt]];
}
Node query(int l,int r,int rt,int L,int R) {
if (L <= l && r <= R) return T[rt];
int mid = (l + r) >> ;
if (R <= mid) return query(l, mid, ls[rt], L, R);
else if (L > mid) return query(mid + , r, rs[rt], L, R);
else return query(l, mid, ls[rt], L, R) + query(mid + , r, rs[rt], L, R);
}
bool check(int x,int l1,int r1,int l2,int r2) {
Node a, b, c; b.sum = ;
if (l2 > r1 + ) b = query(, n, Root[x], r1 + , l2 - );
a = query(, n, Root[x], l1, r1);
c = query(, n, Root[x], l2, r2);
return (a.Rmx + b.sum + c.Lmx) >= ;
}
int main() {
n = read();
for (int i = ; i <= n; ++i) {
A[i].first = read(), A[i].second = i;
}
sort(A + , A + n + );
build(, n, Root[]);
for (int i = ; i <= n; ++i)
update(, n, Root[i], Root[i - ], A[i - ].second);
int c[], m = read(), ans = , pos;
while (m --) {
for (int i = ; i < ; ++i) c[i] = (read() + ans) % n + ;
sort(c, c + );
int l = , r = n;
while (l <= r) {
int mid = (l + r) >> ;
if (check(mid, c[], c[], c[], c[])) pos = mid, l = mid + ;
else r = mid - ;
}
ans = A[pos].first;
printf("%d\n", ans);
}
return ;
}

2653: middle的更多相关文章

  1. bzoj 2653: middle (主席树+二分)

    2653: middle Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2522  Solved: 1434[Submit][Status][Disc ...

  2. 【BZOJ】2653: middle

    2653: middle Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2381  Solved: 1340[Submit][Status][Disc ...

  3. [BZOJ 2653] middle(可持久化线段树+二分答案)

    [BZOJ 2653] middle(可持久化线段树+二分答案) 题面 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整. 给你一个长度为n的序 ...

  4. BZOJ 2653 middle

    AC通道:http://www.lydsy.com/JudgeOnline/problem.php?id=2653 题目大意:多组询问,求左右端点在规定范围内移动所能得到的最大中位数. [分析] 求中 ...

  5. BZOJ 2653: middle 主席树 二分

    https://www.lydsy.com/JudgeOnline/problem.php?id=2653 因为是两个方向向外延伸所以不能对编号取前缀和(这里只有前缀和向后传递的性质,不是实际意义的和 ...

  6. BZOJ 2653 middle | 主席树

    题目: http://www.lydsy.com/JudgeOnline/problem.php?id=2653 题解: 设答案为ans,把大于等于ans的记为1,小于的记为-1,这样可以知道当前an ...

  7. bzoj 2653 middle (可持久化线段树)

    middle Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1981  Solved: 1097[Submit][Status][Discuss] D ...

  8. BZOJ 2653: middle [主席树 中位数]

    传送门 题意: 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你一个 长度为n的序列s.回答Q个这样的询问:s的左端点在[a,b]之间,右 ...

  9. bzoj 2653 middle 二分答案 主席树判定

    判断中位数是否可行需要将当前的解作为分界,大于其的置为1,小于为-1,然后b-c必选,ab,cd可不选,这个用线段树判定就好 但不能每次跑,所以套主席树,按权值排序,构建主席树,更新时将上一个节点改为 ...

随机推荐

  1. 向服务器post或者get数据返回

    #region 向服务器端Get值返回 /// <summary> /// 向服务器端Get返回 /// </summary> ///<see cref="Au ...

  2. Django 补充

    在Django项目的外面操作这个Django内部的models: 当你创建Django项目的时候你在用的时候,你是在这个Django项目中使用的  那么你怎么在你的这个Django项目的外面使用这个D ...

  3. SQL Server 常用数据类型

    char:    固定长度,存储ANSI字符,不足的补英文半角空格. varchar:  可变长度,存储ANSI字符,根据数据长度自动变化. nchar:   固定长度存储Unicode字符,汉字英文 ...

  4. [翻译] UIGlossyButton

    UIGlossyButton https://github.com/waterlou/UIGlossyButton Feature create standard iPhone buttons wit ...

  5. [翻译] BTSimpleRippleButton

    BTSimpleRippleButton https://github.com/balram3429/btSimpleRippleButton This is a custom button for ...

  6. 浅析NSTextContainer

    浅析NSTextContainer TextKit中的NSTextContainer有点晦涩难懂,如果想用TextKit实现文本分页的效果,你是必须要使用NSTextContainer的...... ...

  7. 基于NSString处理文件的高级类

    基于NSString处理文件的高级类 我已经把处理文件的类简化到了变态的程度,如果你还有更简洁的方法,请告知我,谢谢! 使用详情: 源码: // // NSString+File.h // Maste ...

  8. Linux fdisk命令详解[主分区/逻辑分区创建]

    fdisk常见命令参数 -b<分区大小>:指定每个分区的大小: -l:列出指定的外围设备的分区表状况: -s<分区编号>:将指定的分区大小输出到标准输出上,单位为区块: -u: ...

  9. Beanstalkd 的理解

    Beanstalkd 的理解 Beanstalkd 是一个轻量级的内存型队列,利用了和Memcache 类似的协议.其官网beanstakkd官网 下方的感谢语说: Many thanks to me ...

  10. gamit安装

    需要准备的文件: 默认已安装好虚拟机和Ubuntu系统 1.输入用户名密码,进入Ubuntu10.04桌面.按下“Ctrl+Alt+T”,进入终端: 2.在终端输入“sudo gedit /etc/a ...