线段树好题! P2824 [HEOI2016/TJOI2016]排序 题解
前言#
线段树好题!!!!
咕咕了挺久的一道题目,很早之前就想写了,今天终于找了个时间A掉了。
题意#
给定一个
1.0 l r
表示将下标在
2.1 l r
表示将下标在
给定一个数
#
看到数据范围,发现前
注意到
考虑对一段
但是原序列不是
可以选取一个基准数,让原序列大于等于这个数的都变成
如果操作完之后
总复杂度
code#
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1e5 + 5, INF = 0x3f3f3f3f;
const ll mod = 1e9 + 7;
int n, m, pos;
int a[N];
struct question
{
int op, l, r;
} Q[N];
struct segment_tree
{
int l, r, val, tag;
#define l(x) tr[x].l
#define r(x) tr[x].r
#define val(x) tr[x].val
#define tag(x) tr[x].tag
} tr[N << 2];
void pushup(int x)
{
val(x) = val(x << 1) + val(x << 1 | 1);
}
void pushdown(int x)
{
if(tag(x) == -1) return;
val(x << 1) = (r(x << 1) - l(x << 1) + 1) * tag(x);
tag(x << 1) = tag(x);
val(x << 1 | 1) = (r(x << 1 | 1) - l(x << 1 | 1) + 1) * tag(x);
tag(x << 1 | 1) = tag(x);
tag(x) = -1;
}
void build(int l, int r, int x, int v)
{
l(x) = l, r(x) = r, tag(x) = -1, val(x) = 0;
if(l == r)
{
val(x) = (a[l] >= v);
return;
}
int mid = l + r >> 1;
build(l, mid, x << 1, v), build(mid + 1, r, x << 1 | 1, v);
pushup(x);
}
void update(int l, int r, int x, int v)
{
if(l <= l(x) && r(x) <= r)
{
tag(x) = v;
val(x) = (r(x) - l(x) + 1) * v;
return;
}
pushdown(x);
int mid = l(x) + r(x) >> 1;
if(l <= mid) update(l, r, x << 1, v);
if(r > mid) update(l, r, x << 1 | 1, v);
pushup(x);
}
int query(int l, int r, int x)
{
if(l <= l(x) && r(x) <= r) return val(x);
pushdown(x);
int mid = l(x) + r(x) >> 1, res = 0;
if(l <= mid) res += query(l, r, x << 1);
if(r > mid) res += query(l, r, x << 1 | 1);
return res;
}
int check(int v)
{
build(1, n, 1, v);
for(int i = 1;i <= m;i ++)
{
int l = Q[i].l, r = Q[i].r, op = Q[i].op;
int sum = query(l, r, 1);
if(sum == 0) continue;
update(l, r, 1, 0);
if(op == 0) update(r - sum + 1, r, 1, 1);
else update(l, l + sum - 1, 1, 1);
}
return query(pos, pos, 1);
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(nullptr);
cin >> n >> m;
for(int i = 1;i <= n;i ++) cin >> a[i];
for(int i = 1;i <= m;i ++) cin >> Q[i].op >> Q[i].l >> Q[i].r;
cin >> pos;
int l = 1, r = n, res;
while(l <= r)
{
int mid = l + r >> 1;
if(check(mid)) l = mid + 1, res = mid;
else r = mid - 1;
}
cout << res << '\n';
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 单线程的Redis速度为什么快?
· SQL Server 2025 AI相关能力初探
· AI编程工具终极对决:字节Trae VS Cursor,谁才是开发者新宠?
· 展开说说关于C#中ORM框架的用法!