- 作者:老汪软件技巧
- 发表时间:2024-09-23 10:01
- 浏览量:
引言
线段树(Segment Tree)是一种常见的数据结构,用于解决一些线性区间的修改、查询问题,比如对于问题:
如果该问题使用正常的数组方式来遍历求解,假设该数组长度为 N,每次修改和查询的操作耗时是 O(N) 的;但线段树经过 O(N log N) 的构建之后,可以对上述两个操作做到 O(log N) 的优秀复杂度,足以体现其在区间问题上的重要性。
当然上面的例子只是线段树可以解决的一个简单问题,它可以做到的更复杂、更有趣的事情还有很多。在接下来的几篇文章当中我们将会学习使用线段树的概念以及如何使用 MoonBit 实现它,最终我们将一步步实现一棵支持区间加法与乘法、并可以查询区间和、拥有不可变特性的线段树。
本节我们将学习线段树的基本原理以及如何使用 MoonBit 编写一棵最基本的支持单点修改、查询的线段树。
线段树是什么?
本节是纯粹的概念、理论内容,如果读者已经了解并且熟悉线段树的构成与其原理,可以直接阅读下一节内容。
就像引言当中所说的,线段树可以解决一类区间问题,但他长什么样子,能做到如此优秀复杂度的原理又是什么呢?
我们以下图一个线性的数字序列为例,如果我们希望以它建立一棵线段树,那么它将会长这个样子:
可以看到我们把一个线性序列的区间层层分而治之,每次分割为两个对等(如果是奇数则一边多一个)的两个区间(区间范围下标在图示中),最终直到分割为长度为一的区间,并且在此过程中计算了其中每个区间元素的加和(在括号中),这样就从一个线性序列创建了一棵支持查询区间和线段树。
那么在查询区间和的时候,它如何工作呢?我们以查询区间 1-6 的和为例:
可以发现图中的标红部分加起来就等于区间 1-6 的区间和,而我们并没有统计到所有元素,只是选取了最少的区间来频出我们需要求解的区间,并且把我们要求的结果(此处为和)合并起来即可得到最终解。而我们只需要从上到下遍历这棵线段树来判断区间之间的交集/包含关系即可选择到符合条件的区间。
具体来说:
根据二进制分解的知识,我们对任何长度为 N 的区间最多只会求解 Log N 个区间,因此复杂度是可以保证的。
这里仅聊到查询操作,关于线段树上的修改操作的原理和实现我们将会在下一节详细说明。
实现基础定义
我们采用一个非常经典的方法来表达线段树:
enum Node {
Nil
Node(Int, Node, Node)
} derive(Show)
其中 Nil 代表空树,而一个 Node 则包含一个它所储存的数据(为 Int 类型)和左右两个节点。
此外,我们还为他 derive 了 Show Trait,我们在遇到问题时可以直接输出这棵树来调试,这样非常直观且方便。
建树
建树是指将一个线性序列抽象为线段树的过程,一般将其称为 build。
作为前置,我们应该根据需求为 Node 类型编写一个 op_add 的重载来配合下面建树的过程:
fn op_add(self : Node, v : Node) -> Node {
match (self, v) {
(Node(left, _, _), Node(right, _, _)) => Node(left + right, self, v)
(Node(_), Nil) => self
(Nil, Node(_)) => v
(Nil, Nil) => Nil
}
}
定义这一运算之后就可以轻松的向上合并两个 Node 节点,并在此过程中维护区间的和,为我们建树打下了基础,在有些线段树的叙述当中,这个过程也叫做 pushup。
我们可以用 MoonBit 的 ArrayView 特性(某些语言当中也叫做 slice)作为参数来低成本的取出一个线性结构的一段进行递归建树,这个过程是 O(N Log N) 的:
fn build(data : ArrayView[Int]) -> Node {
if data.length() == 1 {
Node(data[0], Nil, Nil)
} else {
let mid = (data.length() + 1) >> 1
build(data[0:mid]) + build(data[mid:])
}
}
分析一下这段代码:
这段代码是非常简洁、可读性非常高的,而且对优化非常友好,可以作为后续其他数据结构的范式学习。
让我们来建立一棵树并输出看看:
fn main {
let tree = build([1, 2, 3, 4, 5][:])
println(tree)
}
运行后的输出是:
Node(15, Node(6, Node(3, Node(1, Nil, Nil), Node(2, Nil, Nil)), Node(3, Nil, Nil)), Node(9, Node(4, Nil, Nil), Node(5, Nil, Nil)))
漂亮,我们已经成功完成了建树的过程!
查询
接下来我们要编写查询,因为这棵线段树的节点向上合并时维护的是区间和,因此我们可以编写一个 query 函数来查询它:
let empty_node : Node = Node(0, Nil, Nil)
fn query(self : Node, l : Int, r : Int, query_l : Int, query_r : Int) -> Node {
if query_l > r || l > query_r {
empty_node
} else if query_l <= l && query_r >= r {
self
} else {
let Node(_, left, right) = self
let mid = (l + r) >> 1
left.query(l, mid, query_l, query_r) +
right.query(mid + 1, r, query_l, query_r)
}
}
首先,l与r是当前函数中已经查询到的区间,query_l与query_r是需要查询的区间,让我们来尝试解析一下这段实现:
Questions and Anwsers:
让我们来测试一下这个查询过程:
fn main {
let tree = build([1, 2, 3, 4, 5][:])
let sum = match tree.query(1, 5, 1, 3) {
Node(sum, _, _) => sum
_ => panic()
}
println(sum)
}
输出是:
6
太好了,我们得到了正确的输出!
代码
完整代码见下
enum Node {
Nil
Node(Int, Node, Node)
} derive(Show)
let empty_node : Node = Node(0, Nil, Nil)
fn op_add(self : Node, v : Node) -> Node {
match (self, v) {
(Node(left, _, _), Node(right, _, _)) => Node(left + right, self, v)
(Node(_), Nil) => self
(Nil, Node(_)) => v
(Nil, Nil) => Nil
}
}
fn build(data : ArrayView[Int]) -> Node {
if data.length() == 1 {
Node(data[0], Nil, Nil)
} else {
let mid = (data.length() + 1) >> 1
build(data[0:mid]) + build(data[mid:])
}
}
fn query(self : Node, l : Int, r : Int, query_l : Int, query_r : Int) -> Node {
if query_l > r || l > query_r {
empty_node
} else if query_l <= l && query_r >= r {
self
} else {
let Node(_, left, right) = self
let mid = (l + r) >> 1
left.query(l, mid, query_l, query_r) +
right.query(mid + 1, r, query_l, query_r)
}
}
fn main {
let tree = build([1, 2, 3, 4, 5][:])
let sum = match tree.query(1, 5, 1, 3) {
Node(sum, _, _) => sum
_ => panic()
}
println(sum)
}
总结
今天我们学习了如何编写一棵简单的线段树的构建和查询操作的编写,下一节课我们将会学习更加复杂的线段树的原理和实现,感兴趣的读者可以在阅读文章之后自行实现下面内容来巩固知识和拓展更多内容: