线段树
线段树
基础知识
以查询区间最大值为例:
由分治的思想可知:区间 \([L,R]\)
的最大值等于 \(\max([L, M], [M + 1,
R])\)
在区间 \([1,n]\) 上建线段树,以区间
\([1, 10]\) 为例,如图:
其中每一个节点里存的都是该节点管辖区间内(黄色区域)的最小值,图中共有 \(2n-1 = 19\) 个节点
build (静态开点)
1 | struct segment_tree |
query
设 \([L,R]\) 是要查询的区间,\([l,r]\) 是线段树的节点区间,\(s_l\) 为 \([l,r]\) 的左儿子区间,\(s_r\) 为区间 \([l,r]\) 的右儿子区间 1. \([l,r] \in [L,R]\) ,直接返回 \([l,r]\) 节点存的值 2. \([L,R] \cap s_l \neq \varnothing\) ,递归处理左儿子 3. \([L,R] \cap s_r \neq \varnothing\) ,递归处理右儿子
例如对上图线段树查询区间 \([5,9]\) 的最小值的情况如下:
其中,蓝色区间是进行了递归查询左/右儿子的区间,红色区间是直接可以获取到值的区间,黄色是没有被访问到的区间
时间复杂度约 \(O(4log n)\)
1
2
3
4
5
6
7
8
9int query(int u, int l, int r)
{
if(l <= t[u].l && t[u].r <= r) return t[u].val;
int mid = t[u].l + t[u].r >> 1;
int ans = 0;
if(l <= mid) ans = query(u << 1, l, r);
if(r > mid) ans = max(ans, query(u << 1 | 1, l, r));
return ans;
}
pushup
用子节点的信息更新父节点的信息 1
2
3
4
5void pushup(int u)
{
t[u].val = max(t[u << 1].val, t[u << 1 | 1].val);
// 也可以采用重载 + 运算符
}
modify
1
2
3
4
5
6
7
8
9
10
11
12
13
14
void modify(int u, int x, int d)
{
// if(l <= t[u].l && t[u].r <= r) // 区间修改
if(t[u].l == x && t[u].r == x) // 单点修改
t[u].val = d;
else
{
int mid = t[u].l + t[u].r >> 1;
// if(mid >= l)
if(mid >= x) modify(u << 1, x, d);
else modify(u << 1 | 1, x, d);
pushup(u);
}
}
1 | void modify(int u, int x, int d) |
以下讲解待补充(
线段树的可持久化(主席树)维护第 \(k\) 大
1 | const int N = 100010, M = 10010; |
Lazy Tag
维护序列
区间加、区间乘,问区间和 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105typedef long long ll;
const int N = 1e5 + 2;
int p, n;
struct node {
int l, r;
ll sum;
ll add = 0, mul = 1;
}t[N * 4];
int a[N];
void pushup(int u)
{
t[u].sum = t[u << 1].sum + t[u << 1 | 1].sum;
t[u].sum %= p;
}
void build(int u, int l, int r)
{
if (l == r)
{
t[u] = {l, r, a[r], 0, 1};
//printf("build : t[%d].sum = %lld\n", u, t[u].sum);
return;
}
int mid = l + r >> 1;
t[u] = { l,r };
build(u << 1, l, mid);
build(u << 1 | 1, mid + 1, r);
pushup(u);
}
// 统一操作,先乘再加!!!!
void calculate(node& t, ll add, ll mul) // 多次操作写成一个函数简洁一些,记得加引用QAQ!!!
{
t.mul = (t.mul * mul)%p; // 多取模()
t.add = (t.add * mul + add)%p;
t.sum = (t.sum * mul + (t.r- t.l + 1)*add)%p;
}
// 将父节点的信息更新给子节点,同时清除父节点上的 tag 标记
void pushdown(int u)
{
calculate(t[u<<1],t[u].add,t[u].mul);
calculate(t[u<<1|1],t[u].add, t[u].mul);
t[u].add = 0;
t[u].mul = 1;
}
void modify(int u, int l, int r, int c,int d)
{
if(t[u].l >= l && t[u].r <= r)
{
calculate(t[u],d,c);
return;
}
// 分裂修改时一定要先 pushdown
pushdown(u);
int mid = t[u].l + t[u].r >>1;
if(mid >= l)modify(u << 1, l, r, c, d);
if(mid < r)modify(u << 1 | 1, l, r, c, d);
pushup(u);
}
ll ask(int u, int l, int r)
{
if (t[u].l >= l && t[u].r <= r)
{
//printf("t[%d].sum = %lld\n",u,t[u].sum);
return t[u].sum;
}
pushdown(u);
ll ans = 0;
if (t[u << 1].r >= l)ans = ask(u << 1, l, r);
ans %= p;
if (t[u << 1 | 1].l <= r)ans += ask(u << 1 | 1, l, r);
ans %= p;
return ans;
}
int main()
{
cin >> n >> p;
for (int i = 1; i <= n; i++)cin >> a[i];
build(1, 1, n);
int m;cin >> m;
int op, l, r, d;
while (m--)
{
cin >> op >> l >> r;
if (op != 3)
{
cin >> d;
if(op == 1)
modify(1, l, r, d, 0);
else
modify(1, l, r, 1, d);
}
else {
cout << ask(1, l, r) << '\n';
}
}
return 0;
}
线段树合并
技能树待点亮(