01二分 [AGC006D] Median Pyramid Hard + P2824 [HEOI2016/TJOI2016] 排序
[AGC006D] Median Pyramid Hard
考虑对于一个长度为 2n + 1 的 01 序列 b 如何快速确定堆顶元素。
_ _ _ _ x
_ _ _ 0 x
_ _ 0 0 x
_ x 0 0 x
x x 0 0 x
容易得到,两个相同元素能够一直往上走,直到边界。
如果有两个相同元素出现在第 n 位,那答案显然为 b[n]。
那如果不在呢?
_ _ _ _ _ 0 0 x
_ _ _ _ 0 0 1 x
_ _ _ 0 0 1 0 x
_ _ 0 0 1 0 1 x
_ 0 0 1 0 1 0 x
0 0 1 0 1 0 1 x
得到结论,交替出现的 01 段能让两个相同元素斜向上平移。
由于离中心越近,越快能平移到中心。
因此离中心最近的两个相同元素即为答案。
最后特判没有相同元素的情况,也就是 01 交替出现,显然答案为 b[1]。
回到原题,考虑二分答案。
把原序列中 >= mid 的设为 1,< mid 的设为 0。
如果堆顶为 1 说明 ans >= mid,否则 < mid。
#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i <= (b); ++ i)
#define per(i, a, b) for(int i = (a); i >= (b); -- i)
#define pb emplace_back
#define All(X) X.begin(), X.end()
using namespace std;
using ll = long long;
constexpr int N = 2e5 + 5;
int n, m, a[N], b[N];
int l, r, mid;
bool check(int v) {
rep(i, 1, m) b[i] = a[i] >= v;
rep(i, 0, n - 2) {
if(b[n + i] == b[n + i + 1]) return b[n + i];
if(b[n - i] == b[n - i - 1]) return b[n - i];
}
return b[1];
}
int main() {
ios::sync_with_stdio(false), cin.tie(nullptr), cout.tie(nullptr);
cin >> n; m = n * 2 - 1;
rep(i, 1, m) cin >> a[i];
l = 1, r = m;
while(l < r) {
if(check(mid = l + r + 1 >> 1)) l = mid;
else r = mid - 1;
}
cout << l;
return 0;
}
P2824 [HEOI2016/TJOI2016] 排序
题意:对一个长度为 n 的排列排 m 次序,问最终 p 位置是哪个数。
考虑对一个 01 序列进行一次排序需要多长时间。
令 cnt 等于 [l,r] 内 1 的个数。
对于升序操作,
- 将 [l,r - cnt] 全改为 0。
- 将 [r - cnt + 1, r] 全改为 1。
降序同理。
用线段树能做到 log n。
二分答案。
把原序列中 >= mid 的设为 1,< mid 的设为 0。
进行 m 次排序。
如果目标位置为 1 说明 ans >= mid,否则 < mid。
总复杂度 \(O(m\log^2n)\)。
#include<bits/stdc++.h>
#define rep(i, a, b) for(int i = (a); i <= (b); ++ i)
#define per(i, a, b) for(int i = (a); i >= (b); -- i)
#define pb emplace_back
#define All(X) X.begin(), X.end()
using namespace std;
using ll = long long;
constexpr int N = 1e5 + 5;
struct Node {
int op, l, r;
} q[N];
struct Seg {
int v, tag, sz;
} t[N << 2];
int n, m, pos, a[N];
void pushup(int x) {
t[x].v = t[x << 1].v + t[x << 1 | 1].v;
}
void pushdown(int x) {
if(~t[x].tag) {
int l = x << 1, r = l | 1;
t[l].v = t[l].sz * t[x].tag, t[l].tag = t[x].tag;
t[r].v = t[r].sz * t[x].tag, t[r].tag = t[x].tag;
t[x].tag = -1;
}
}
void build(int v, int x = 1, int l = 1, int r = n) {
t[x] = {0, -1, r - l + 1};
if(l == r) {
t[x].v = a[l] >= v;
return;
}
int mid = l + r >> 1;
build(v, x << 1, l, mid);
build(v, x << 1 | 1, mid + 1, r);
pushup(x);
}
int query(int L, int R, int x = 1, int l = 1, int r = n) {
if(L > R) return 0;
if(L <= l && r <= R) return t[x].v;
int mid = l + r >> 1, ret = 0;
pushdown(x);
if(mid >= L) ret += query(L, R, x << 1, l, mid);
if(mid < R) ret += query(L, R, x << 1 | 1, mid + 1, r);
return ret;
}
void modify(int L, int R, int v, int x = 1, int l = 1, int r = n) {
if(L > R) return;
if(L <= l && r <= R) return t[x].v = t[x].sz * (t[x].tag = v) , void();
int mid = l + r >> 1;
pushdown(x);
if(mid >= L) modify(L, R, v, x << 1, l, mid);
if(mid < R) modify(L, R, v, x << 1 | 1, mid + 1, r);
pushup(x);
}
bool check(int v) {
build(v);
rep(i, 1, m) {
auto [op, l, r] = q[i];
int cnt = query(l, r);
if(op) {
modify(l, l + cnt - 1, 1);
modify(l + cnt, r, 0);
}
else {
modify(r - cnt + 1, r, 1);
modify(l, r - cnt, 0);
}
}
return query(pos, pos);
}
int main() {
ios::sync_with_stdio(false), cin.tie(nullptr), cout.tie(nullptr);
cin >> n >> m;
rep(i, 1, n) cin >> a[i];
rep(i, 1, m) cin >> q[i].op >> q[i].l >> q[i].r;
cin >> pos;
int l = 1, r = n;
while(l < r) {
int mid = l + r + 1 >> 1;
if(check(mid)) l = mid;
else r = mid - 1;
}
cout << l;
return 0;
}
01二分 [AGC006D] Median Pyramid Hard + P2824 [HEOI2016/TJOI2016] 排序的更多相关文章
- [Luogu P2824] [HEOI2016/TJOI2016]排序 (线段树+二分答案)
题面 传送门:https://www.luogu.org/problemnew/show/P2824 Solution 这题极其巧妙. 首先,如果直接做m次排序,显然会T得起飞. 注意一点:我们只需要 ...
- 洛谷 P2824 [HEOI2016/TJOI2016]排序 解题报告
P2824 [HEOI2016/TJOI2016]排序 题意: 有一个长度为\(n\)的1-n的排列\(m\)次操作 \((0,l,r)\)表示序列从\(l\)到\(r\)降序 \((1,l,r)\) ...
- 洛谷$P2824\ [HEOI2016/TJOI2016]$ 排序 线段树+二分
正解:线段树+二分 解题报告: 传送门$QwQ$ 昂着题好神噢我$jio$得$QwQQQQQ$,,, 开始看到长得很像之前考试题的亚子,,,然后仔细康康发现不一样昂$kk$,就这里范围是$[1,n]$ ...
- P2824 [HEOI2016/TJOI2016]排序
题面 这是一道非常巧妙的线段树的题 我们会发现维护\(1 \sim n\)的序列非常困难,但如果我们维护\(01\)序列的的顺序,就非常容易了 但是我们怎么能把这道题变成维护\(01\)序列的顺序呢? ...
- [洛谷P2824][HEOI2016/TJOI2016]排序
题目大意:一个全排列,两种操作: 1. $0\;l\;r:$把$[l,r]$升序排序2. $1\;l\;r:$把$[l,r]$降序排序 最后询问第$k$位是什么 题解:二分答案,把比这个数大的赋成$1 ...
- 洛谷P2824 [HEOI2016/TJOI2016]排序(线段树)
传送门 这题的思路好清奇 因为只有一次查询,我们考虑二分这个值为多少 将原序列转化为一个$01$序列,如果原序列上的值大于$mid$则为$1$否则为$0$ 那么排序就可以用线段树优化,设该区间内$1$ ...
- luogu P2824 [HEOI2016/TJOI2016]排序
题目描述 在2016年,佳媛姐姐喜欢上了数字序列.因而他经常研究关于序列的一些奇奇怪怪的问题,现在他在研究一个难题,需要你来帮助他.这个难题是这样子的:给出一个1到n的全排列,现在对这个全排列序列进行 ...
- 「Luogu P2824 [HEOI2016/TJOI2016]排序」
一道十分神奇的线段树题,做法十分的有趣. 前置芝士 线段树:一个十分基础的数据结构,在这道题中起了至关重要的作用. 一种基于01串的神奇的二分思想:在模拟赛中出现了这道题,可以先去做一下,这样可能有助 ...
- Luogu P2824 [HEOI2016/TJOI2016]排序 线段树+脑子
只会两个$log$的$qwq$ 我们二分答案:设答案为$ans$,则我们把$a[i]<=ans$全部设成$0$,把$a[i]>ans$全部设成$1$,扔到线段树里,这样区间排序(升序)就是 ...
- 洛谷 P2824 [HEOI2016/TJOI2016]排序 (线段树合并)
(另外:题解中有一种思路很高妙而且看上去可以适用一些其他情况的离线方法) 线段树合并&复杂度的简单说明:https://blog.csdn.net/zawedx/article/details ...
随机推荐
- UE4_C++实现TimeLine
主要实现蓝图节点中时间轴的功能. 目前UE提供了两种实现方式,一个是使用FTimeLine其是一个时间轴的结构体:另一种方式是使用UTimeLineComponent,其是一个时间轴组件类.两者内部定 ...
- vue3 快速入门系列 —— 基础
vue3 快速入门系列 - 基础 前面我们已经用 vue2 和 react 做过开发了. 从 vue2 升级到 vue3 成本较大,特别是较大的项目.所以许多公司对旧项目继续使用vue2,新项目则使用 ...
- 实用 Linux 命令 Windos 命令 实例演示 持续更新中
实用 Linux 命令 Windos 命令 实例演示 持续更新中 目录 实用 Linux 命令 Windos 命令 实例演示 持续更新中 Linux 命令 [Command [options] [lo ...
- Manacher小记
目录 前言 洛谷 3805[模板]manacher算法 题目 分析 代码 洛谷 4555 最长双回文串 题目 分析 代码 洛谷 1659 拉拉队排练 题目 分析 代码 前言 皆移植于原csdn博客,略 ...
- Jetty的http模块
启用http模块,执行如下命令: java -jar $JETTY_HOME/start.jar --add-modules=http 查看http模块的配置文件,执行如下命令: cat $JETTY ...
- 修复HTTP动词篡改导致的认证旁路问题的方法
本文于2016年4月完成,发布在个人博客网站上. 考虑个人博客因某种原因无法修复,于是在博客园安家,之前发布的文章逐步搬迁过来. 诡异的问题 分析AppScan扫描报告的时候,发现报告里提示" ...
- 网络协议之:haproxy的Proxy Protocol代理协议
目录 简介 Proxy Protocol的实现细节 版本1 版本2 Proxy Protocol的使用情况 总结 简介 代理大家应该都很熟悉了,比较出名的像是nginx,apache HTTPD,st ...
- 【直播回顾】参与文档贡献,开启OpenHarmony社区贡献
5月25日晚上19点,战"码"先锋第二期直播 <参与文档贡献,开启OpenHarmony社区贡献> ,在OpenHarmony社群内成功举行. 本期课程,由华为 ...
- Java基础知识:面试官必问的问题
数据类型 基本类型 byte/8 char/16 short/16 int/32 float/32 long/64 double/64 boolean/~ boolean 只有两个值:true.fal ...
- 并发编程面试必备之ConcurrentHashMap源码解析
ConcurrentHashMap在我的面试生涯中,10次有8次是会被问到的,记得刚毕业那会,被问到ConcurrentHashMap源码的无助与苦涩,无奈只能网上找了一些教程,背一背,才算是蒙混过关 ...