\(\text{Solution}\)

不考虑起点区间和终点区间的限制,求区间中位数

可以二分中位数,大于等于中位数的位置赋为 \(1\),小于的位置赋 \(-1\)

当区间和大于等于 \(0\) 时此数才可能为中位数

因为有多个询问,但中位数数值只可能有 \(n\) 个

所以预处理时枚举当前中位数,处理出序列此时区间和的情况,线段树即可

但一棵线段树空间是 \(O(n log n)\) 的,\(n\) 棵不可行

注意到中位数 \(m\) 到 \(m+1\) 时只有值为 \(m\) 的位置从 \(1\) 变到了 \(-1\),其他都一样

启示我们可以用主席树维护,于是这个问题就解决了

回到本题,仍旧预处理并且二分答案

起点区间和终点区间夹的区间是必选的,取出区间和即可

起点和终点待定,确定某个起点后,此起点到起点区间右端点的数都要选,终点同理,即取后缀和前缀

那么在二分中位数的情况下,前缀和后缀越大越好

维护区间和时顺便维护区间最大前后缀即可

\(\text{Code}\)

#include <cstdio>
#include <algorithm>
#define re register
using namespace std; const int N = 20005, INF = N * 10;
int n, m, q[4], rt[N], size;
struct nod{int v, id;}a[N];
inline bool cmp(nod a, nod b){return a.v < b.v;} struct node{int sum, lx, rx;};
struct tree{int ls, rs; node t;}seg[N * 61];
inline node operator + (const node &a, const node &b)
{
return node{a.sum + b.sum, max(a.lx, a.sum + b.lx), max(b.rx, b.sum + a.rx)};
} void update(int &p, int pre, int l, int r, int x, int v)
{
p = ++size, seg[p] = seg[pre];
if (l == r)
{
seg[p].t.sum += v, seg[p].t.lx += v, seg[p].t.rx += v;
return;
}
int mid = l + r >> 1;
if (x <= mid) update(seg[p].ls, seg[pre].ls, l, mid, x, v);
else update(seg[p].rs, seg[pre].rs, mid + 1, r, x, v);
seg[p].t = seg[seg[p].ls].t + seg[seg[p].rs].t;
} node query(int p, int l, int r, int tl, int tr)
{
if (tl > r || tr < l) return node{0, -INF, -INF};
if (tl <= l && r <= tr) return seg[p].t;
int mid = l + r >> 1; node res = {0, -INF, -INF};
if (tl <= mid) res = query(seg[p].ls, l, mid, tl, tr);
if (tr > mid) res = res + query(seg[p].rs, mid + 1, r, tl, tr);
return res;
} inline int check(int mid)
{
int res = 0;
if (q[1] + 2 <= q[2]) res = query(rt[mid], 1, n, q[1] + 1, q[2] - 1).sum;
res += query(rt[mid], 1, n, q[0], q[1]).rx + query(rt[mid], 1, n, q[2], q[3]).lx;
return res >= 0;
} int main()
{
freopen("LG2839.in", "r", stdin), freopen("LG2839.out", "w", stdout);
scanf("%d", &n);
for(re int i = 1; i <= n; i++) scanf("%d", &a[i].v), a[i].id = i;
sort(a + 1, a + n + 1, cmp);
for(re int i = 1; i <= n; i++) update(rt[1], rt[1], 1, n, i, 1);
for(re int i = 2; i <= n; i++) update(rt[i], rt[i - 1], 1, n, a[i - 1].id, -2); scanf("%d", &m);
for(int l, r, mid, ans, lst = 0; m; --m)
{
for(int i = 0; i < 4; i++) scanf("%d", &q[i]), q[i] = (q[i] + lst) % n + 1;
sort(q, q + 4), l = 1, r = n, ans = 0;
while (l <= r)
{
mid = l + r >> 1;
if (check(mid)) ans = mid, l = mid + 1;
else r = mid - 1;
}
printf("%d\n", lst = a[ans].v);
}
}

LG P2839 [国家集训队]middle的更多相关文章

  1. P2839 [国家集训队]middle

    P2839 [国家集训队]middle 好妙的题啊,,,, 首先二分一个答案k,把数列里>=k的数置为1,=0就是k>=中位数,<0就是k<中位数 数列的最大和很好求哇 左边的 ...

  2. [洛谷P2839][国家集训队]middle

    题目大意:给你一个长度为$n$的序列$s$.$Q$个询问,问在$s$中的左端点在$[a,b]$之间,右端点在$[c,d]$之间的子段中,最大的中位数. 强制在线. 题解:区间中位数?二分答案,如果询问 ...

  3. 洛谷P2839 [国家集训队]middle 主席树_二分

    Code: #include <cstdio> #include <algorithm> #include <cstring> #include <strin ...

  4. Luogu P2839 [国家集训队]middle

    题目 首先我们考虑解决中位数一类问题的常用手段:二分\(mid\),将大于等于它的设为\(1\),小于它的设为\(−1\),判断区间和是否\(\ge0\). 对于询问\(a,b,c,d\),二分完\( ...

  5. [国家集训队]middle 解题报告

    [国家集训队]middle 主席树的想法感觉挺妙的,但是这题数据范围这么小,直接分块草过去不就好了吗 二分是要二分的,把\(<x\)置\(-1\),\(\ge x\)的置\(1\),于是我们需要 ...

  6. [国家集训队]middle

    [国家集训队]middle 题目 解法 开\(n\)颗线段树,将第\(i\)颗线段树中大于等于第\(i\)小的数权值赋为1,其他的则为-1,对于每个区间维护一个区间和,最大前缀和,最大后缀和. 然后二 ...

  7. CF484E Sign on Fence && [国家集训队]middle

    CF484E Sign on Fence #include<bits/stdc++.h> #define RG register #define IL inline #define _ 1 ...

  8. 【LG2839】[国家集训队]middle

    [LG2839][国家集训队]middle 题面 洛谷 题解 按照求中位数的套路,我们二分答案\(mid\),将大于等于\(mid\)的数设为\(1\),否则为\(-1\). 若一个区间和大于等于\( ...

  9. BZOJ.2653.[国家集训队]middle(可持久化线段树 二分)

    BZOJ 洛谷 求中位数除了\(sort\)还有什么方法?二分一个数\(x\),把\(<x\)的数全设成\(-1\),\(\geq x\)的数设成\(1\),判断序列和是否非负. 对于询问\(( ...

  10. luogu2839 [国家集训队]middle

    题目链接:洛谷 题目大意:给定一个长度为$n$的序列,每次询问左端点在$[a,b]$,右端点在$[c,d]$的所有子区间的中位数的最大值.(强制在线) 这里的中位数定义为,对于一个长度为$n$的序列排 ...

随机推荐

  1. label studio 结合 MMDetection 实现数据集自动标记、模型迭代训练的闭环

    前言 一个 AI 方向的朋友因为标数据集发了篇 SCI 论文,看着他标了两个多月的数据集这么辛苦,就想着人工智能都能站在围棋巅峰了,难道不能动动小手为自己标数据吗?查了一下还真有一些能够满足此需求的框 ...

  2. MySQL进阶实战4,MySQL索引详解,下篇

    一.索引 索引是存储引擎用于快速查找记录的一种数据结构.我觉得数据库中最重要的知识点,就是索引. 存储引擎以不同的方式使用B-Tree索引,性能也各有不同,各有优劣.例如MyISAM使用前缀压缩技术使 ...

  3. python中的字符串(1)

    1.大小写的转换 upper()/lower() 转成大写.upper() 转成小写.lower() 返回的是字符串 2.是否是数字 isdigit() 返回的布尔值 3.去除字符串的空白字符 str ...

  4. 解决aspnetcore-browser-refresh.js:234 WebSocket connection to 'wss://localhost:62356/Admin/' failed问题

    前言 前段时间升级了Visual Studio到v17.1.1最新版本,然后今天来运行之前的一个.net5项目一直提示:aspnetcore-browser-refresh.js:234 WebSoc ...

  5. rate-limit 一款 java 开源渐进式分布式限流框架使用介绍

    项目简介 rate-limit 是一个为 java 设计的渐进式限流工具. 目的是为了深入学习和使用限流,后续将会持续迭代. 特性 渐进式实现 支持独立于 spring 使用 支持整合 spring ...

  6. PTA散列表平方探测法解决冲突

    PTA散列表平方探测法解决冲突 核心问题   当所有的位置都被填上了,且不能插入关键词,要进入死循环了怎么办? 题目   本题的任务很简单:将给定的无重复正整数序列插入一个散列表,输出每个输入的数字在 ...

  7. $_GET方法踩坑

    背景 写代码时,遇到一个奇怪的问题:小程序卡券解码接口老是报解码失败,苦寻了一个小时,发现原来是url参数经过thinkphp的I方法被过滤掉,而且涉及到PHP原生的$_GET 原因 I方法底层是原生 ...

  8. 基于MongoDb的事件订阅实现hook监听

    详情请参考原文:-- 基于MongoDb的事件订阅实现hook监听(insert,update,remove,find等事件开始,事件成功等)

  9. python进阶之路8 字典、元组、集合内置方法 编码理论

    内容回顾 作业讲解 1.前期不熟练的情况下一定要先写注释 2.一定要仔细思考每一行代码的含义 3.自己不会的代码或者不熟练的代码一定要多敲多练 数据类型内置方法简介 所有的数据类型基本上都自带了一些操 ...

  10. 激光炸弹【算法竞赛进阶指南, HNOI2003】

    激光炸弹 地图上有 \(N\) 个目标,用整数 \(Xi,Yi\)表示目标在地图上的位置,每个目标都有一个价值 \(Wi\). 注意:不同目标可能在同一位置. 现在有一种新型的激光炸弹,可以摧毁一个包 ...