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 ...
随机推荐
- verilog之锁存器和触发器
verilog锁存器和触发器 1.基本概念 锁存,就是输入信号变化时,输出不发生变化时,就是触发器或者锁存器.触发器的敏感信号是clk,即触发器是知道被延时了多少.对于锁存器来说,延时是不确定的.一般 ...
- opensips的dispatcher模块笔记
操作系统 :CentOS 7.6_x64 opensips版本:2.4.9 dispatcher模块模块实现了基于目的地址的调度功能,可用作无状态负载均衡,但不能保证均匀分配.今天整理下CentOS7 ...
- 【已解决】Exception in thread "main" java.lang.RuntimeException: java.net.ConnectException
没有启动hadoop集群
- CF-937(D,E)
CF-937 在补题-- D 分析 我们发现这些因子都是二进制形式的十进制数,n的范围是$1e5$,16的二进制是$10000$,于是可以枚举1~16,把这些因子预处理出来,对于每个n就枚举因子再作除 ...
- 新零售SaaS架构:客户管理系统架构设计(万字图文总结)
什么是客户管理系统? 客户管理系统,也称为CRM(Customer Relationship Management),主要目标是建立.发展和维护好客户关系. CRM系统围绕客户全生命周期的管理,吸引和 ...
- Sqlite数据库联合查询及表复制等详述
外键:一般在两个表之间要建立关联时候,创建一个列创建 为外键(UserInfos-DeptId),它在另一个表必须是主键(DeptInfos-DeptId) 元素约束:主键约束:主要区别内容相同的行, ...
- Qt数据结构-QString二:QString的arg能不能像Python的format一样使用
常规QString拼接字符串我们是这样写的 QString s = QString("My name is %1, age %2").arg("zhangsan" ...
- HDC2021技术分论坛:跨端分布式计算技术初探
作者:zhengkai,分布式通信首席技术专家 当今的移动应用都向着智能化和多样化方向发展,例如AI辅助,VR/AR应用,沉浸式游戏等.然而现实中的移动设备,因为便携性要求受限于尺寸.电池容量以及温控 ...
- 升级gradle:Could not find method jackOptions() for arguments
前言 这是我在升级gradle发生的错误. 解决 原因是被废弃了: 删除: jackOptions { enabled true } 解释一下什么是jack: Jack 是 Java Android ...
- js es6系列——map函数
正文 map,必要解释就是map不是地图的意思,而是映射的意思. 这里就简单的介绍了这个map了. array.map(callback,[ thisObject]); 看下这个案例后,我们发现了就发 ...