P4097 【模板】李超线段树 / [HEOI2013] Segment
P4097 【模板】李超线段树 / [HEOI2013] Segment
前言
李超线段树并不是一种新的线段树,而是对一类题维护最值的过程做了改进,使线段树仍然有不错的复杂度。
引入
简要题意#
实现两种操作:
- 在区间
上加入一条两端为 , 的线段。 - 查询下标
上的纵坐标最大的线段的编号。
区别如区间覆盖,这里的每个位置加入的都是不同的值,如果暴力,每次修改就要
不妨让线段树上每个节点
现在插入一条线段
假如此时遍历到的区间为
运用标记永久化的思想,大于一半都是同一个编号的区间就不管了,直接把答案挂在
原本
如何判断是否超过?只需要看两个端点上线段的纵坐标即可。
这样,修改操作就在
查询操作简单,由于标记永久化,只要比较线段树上到
总复杂度
#include <iostream>
#define pii std::pair<int, int>
#define mk std::make_pair
#define fi first
#define se second
#define pb push_back
using i64 = long long;
using ull = unsigned long long;
const i64 iinf = 0x3f3f3f3f, linf = 0x3f3f3f3f3f3f3f3f;
const int N = 1e5 + 10, mod1 = 39989, mod2 = 1e9;
int n, cnt;
double eps = 1e-9;
struct line {
double k, b;
} a[N];
int t[N << 2];
int cmp(double x, double y) {
if(x - y > eps) return 1;
if(y - x > eps) return -1;
return 0;
}
double calc(int id, int x) {
return a[id].k * x + a[id].b;
}
void add(int x0, int y0, int x1, int y1) {
if(x0 == x1) {
a[++cnt].k = 0, a[cnt].b = std::max(y0, y1);
} else {
a[++cnt].k = 1.0 * (y1 - y0) / (x1 - x0), a[cnt].b = y0 - a[cnt].k * x0;
}
}
void mdf(int u, int l, int r, int x) {
int mid = (l + r) >> 1;
int bmid = cmp(calc(x, mid), calc(t[u], mid));
if(bmid == 1 || (!bmid && x < t[u])) std::swap(x, t[u]);
int bl = cmp(calc(x, l), calc(t[u], l)), br = cmp(calc(x, r), calc(t[u], r));
if(bl == 1 || (!bl && x < t[u])) mdf(u << 1, l, mid, x);
if(br == 1 || (!br && x < t[u])) mdf(u << 1 | 1, mid + 1, r, x);
}
void update(int u, int l, int r, int L, int R, int x) {
if(L <= l && r <= R) {
mdf(u, l, r, x);
return;
}
int mid = (l + r) >> 1;
if(L <= mid) update(u << 1, l, mid, L, R, x);
if(R > mid) update(u << 1 | 1, mid + 1, r, L, R, x);
}
int mx(int x, int y, int z) {
int ret = cmp(calc(x, z), calc(y, z));
if(ret == 1) return x;
else if(ret == -1) return y;
else return (x < y ? x : y);
}
int qry(int u, int l, int r, int x) {
if(l == r) return t[u];
int mid = (l + r) >> 1;
if(x <= mid) return mx(t[u], qry(u << 1, l, mid, x), x);
else return mx(t[u], qry(u << 1 | 1, mid + 1, r, x), x);
}
int main() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
std::cin >> n;
int lstans = 0;
while(n--) {
int op, x0, y0, x1, y1, k;
std::cin >> op;
if(!op) {
std::cin >> k;
k = (k + lstans - 1) % mod1 + 1;
std::cout << (lstans = qry(1, 1, mod1, k)) << "\n";
} else {
std::cin >> x0 >> y0 >> x1 >> y1;
x0 = (x0 + lstans - 1) % mod1 + 1;
x1 = (x1 + lstans - 1) % mod1 + 1;
y0 = (y0 + lstans - 1) % mod2 + 1;
y1 = (y1 + lstans - 1) % mod2 + 1;
if(x0 > x1) std::swap(x0, x1), std::swap(y0, y1);
add(x0, y0, x1, y1);
update(1, 1, mod1, x0, x1, cnt);
}
}
return 0;
}
标签:
线段树
Buy me a cup of coffee ☕.
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具