Appearance
二分搜索树又称二分排序树,它具有很多优异的性质,能快速完成特定场景下的需求,同时也是一个很好的联系递归思维的数据结构。
- 根节点:最顶层的节点
- 叶子节点:最底层的节点
基本结构
新建一个BST类,二分搜索树的底层是链表,其中子类Node
来表示链表结构,BST维护root
根节点和树的大小。
java
public class BST<E extends Comparable<E>> {
private class Node {
public E e;
public Node left, right;
public Node(E e) {
this.e = e;
left = null;
right = null;
}
}
private Node root;
private int size;
public BST() {
root = null;
size = 0;
}
public int size() {
return size;
}
public boolean isEmpty() {
return size == 0;
}
}
二分搜索树的添加方法
首先来看添加操作,如果树为空则添加根节点,否则进行下一步操作,具体看注释。
java
public class BST<E extends Comparable<E>> {
public void add(E e) {
root = add(root, e);
if (root == null) {
root = new Node(e);
size++;
}else {
add(node,e);
}
}
private void add(Node node, E e) {
if (node.e.equals(e)) {
return; // 1.如果相等,则不进行添加
} else if (e.compareTo(node.e) < 0 && node.left == null) { // 2.如果 e<node.e 且左子树为空,则新建到左子树
node.left = new Node(e);
size++;
} else if (e.compareTo(node.e) > 0 && node.right == null) { // 3.如果 e>node.e 且右子树为空,则新建到叶子节点
node.right = new Node(e);
size++;
} else if (e.compareTo(node.e) < 0) { // 4.如果 e<node.e,则递归调用左子树,直到遍历到叶子节点
add(node.left, e);
} else {
add(node.right, e); // 4.如果 e>node.e,则递归调用右子树,直到遍历到叶子节点
}
}
}
二分搜索树的添加方法改进
上述的方法有没有问题?首先添加操作需要额外对添加根节点进行判断,下面介绍一种更加优雅的方法,不但能解决这个问题,还能使得代码更加简洁。
java
public class BST<E extends Comparable<E>> {
public void add(E e) {
root = add(root, e);
}
// 添加方法改进版
private Node add(Node node, E e) {
if (node == null) {
size++;
return new Node(e);
}
if (e.compareTo(node.e) <= 0) {
node.left = add(node.left, e); //宏观理解方式:对node.left进行重新赋值,这是通过递归得到包含新数据的树
} else if (e.compareTo(node.e) > 0) {
node.right = add(node.right, e);
}
return node;
}
}
是否包含某数据
同样是采用递归实现,如果比node节点大就去右子树找,如果比node节点小就去左子树找,找到最后如果为null就是不包含。
java
public class BST<E extends Comparable<E>> {
public boolean contains(E e) {
return contains(root, e);
}
private boolean contains(Node node, E e) {
if (node == null) {
return false;
}
if (e.compareTo(node.e) == 0) {
return true;
}
if (e.compareTo(node.e) < 0) {
return contains(node.left, e);
} else {
return contains(node.right, e);
}
}
}
二分搜索树的前序遍历
二分搜索树的前序遍历的记忆法则是“根左右",即先遍历根节点,再遍历左子树节点,再遍历右子树节点。递归实现起来很容易,三步即可解决,先输出数据,然后递归左子树,最后递归右子树。
java
public class BST<E extends Comparable<E>> {
public void preOrder() {
preOrder(root);
}
private void preOrder(Node node) {
if (node == null) {
return;
}
System.out.print(node.e+"\t");
preOrder(node.left);
preOrder(node.right);
}
}
二分搜索树的中序与后序遍历
有了前序遍历的基础,中序与后序也很简单,相对而言计算机领域常考的是其遍历结果,不过这不是本篇文章的重点,就不过多赘述了。
java
public class BST<E extends Comparable<E>> {
/** 二分搜索树中序遍历 */
public void inOrder() {
inOrder(root);
System.out.println();
}
private void inOrder(Node node) {
if (node == null) {
return;
}
inOrder(node.left);
System.out.print(node.e+"\t");
inOrder(node.right);
}
/** 二分搜索树后序遍历 */
public void postOrder() {
postOrder(root);
System.out.println();
}
private void postOrder(Node node) {
if (node == null) {
return;
}
postOrder(node.left);
postOrder(node.right);
System.out.print(node.e+"\t");
}
}
注意看中序遍历的数据,发现是有序的。
二分搜索树遍历的测试
java
public static void main(String[] args) {
BST<Integer> bst = new BST<>();
///////////////////////
// 5 //
// / \ //
// 3 6 //
// / \ / \ //
// 2 4 8 //
// / //
// 7 //
///////////////////////
int[] nums = {5, 3, 6, 8, 4, 2, 7};
for (int num : nums) {
bst.add(num);
}
System.out.print("前序遍历:");
bst.preOrder();
System.out.print("中序遍历:");
bst.inOrder();
System.out.print("后序遍历:");
bst.postOrder();
}
测试结果:
shell
前序遍历:5 3 2 4 6 8 7
中序遍历:2 3 4 5 6 7 8
后序遍历:2 4 3 7 8 6 5
重写二分搜索树的toString方法
为了方便测试,这里简单重写下toString方法,将树的深度遍历出来,效果没有上述注释部分表示的明显。
java
public class BST<E extends Comparable<E>> {
@Override
public String toString() {
StringBuffer res = new StringBuffer();
generateBSTString(root, 0, res);
return res.toString();
}
private void generateBSTString(Node node, int depth, StringBuffer res) {
if (node == null) {
res.append(generateDepthString(depth) + "null\n");
return;
}
res.append(generateDepthString(depth) + node.e + "\n");
generateBSTString(node.left, depth + 1, res);
generateBSTString(node.right, depth + 1, res);
}
private String generateDepthString(int depth) {
StringBuffer res = new StringBuffer();
for (int i = 0; i < depth; i++) {
res.append("--");
}
return res.toString();
}
}
效果如下
java
5
--3
----2
------null
------null
----4
------null
------null
--6
----null
----8
------7
--------null
--------null
------null
获取二分搜索树的最小值与最大值
二分搜索树有两个性质,①若它的左子树不为空,则左子树上所有节点的值都小于根节点的值;②若它的右子树不为空,则右子树上所有节点的值都大于根节点的值
根据二分搜索树的性质不难得到,最左侧底部的叶子节点就是二分搜索树的最小值,最右侧底部的叶子节点就是二分搜索树的最大值,根据递归获取到二分搜索树的最小值与最大值。
java
public class BST<E extends Comparable<E>> {
public E minimum() {
if (size == 0){
throw new IllegalArgumentException("当前序列无数据");
}
return minimum(root).e;
}
private Node minimum(Node node) {
if (node.left == null) {
return node;
}
return minimum(node.left);
}
}
上述代码是获取二分搜索树的最小值,下面是获取二分搜索树的最大值
java
public class BST<E extends Comparable<E>> {
public E maximum() {
if (size == 0){
throw new IllegalArgumentException("当前序列无数据");
}
return maximum(root).e;
}
private Node maximum(Node node) {
if (node.right == null) {
return node;
}
return maximum(node.right);
}
}
删除二分搜索树的最大值与最小值
删除最小值不难,由于最小值的左子树一定为空,如果左子树不为空说明它不是最小的,这个不难理解。但是最小值的右子树可能不为空,这样的情况下,我们需要将最小值的右子树根节点代替最小值的位置,这样就完成了删除最小值。
java
public class BST<E extends Comparable<E>> {
public E removeMin() {
// 找到最小值
E ret = minimum();
root = removeMin(root);
return ret;
}
private Node removeMin(Node node) {
// 当子节点的左子树为null时,说明当前节点就是最小值
if (node.left == null) {
// 将要删除节点的右子树,代替删除节点的位置,右子树为空也没有关系,不影响
Node rightNode = node.right;
node.right = null;
size--;
return rightNode;
}
node.left = removeMin(node.left);
return node;
}
}
删除最大值与删除最小值相似,只不过需要考虑最大值左子树不为空的情况。
java
public class BST<E extends Comparable<E>> {
public E removeMax() {
// 找到最大值
E ret = maximum();
root = removeMax(root);
return ret;
}
private Node removeMax(Node node) {
if (node.right == null) {
Node leftNode = node.left;
node.left = null;
size--;
return leftNode;
}
node.right = removeMax(node.right);
return node;
}
}
删除二分搜索树的任意值
有了删除最大值与最小值的基础,被删除的节点无论是左子树为空还是右子树为空,处理方法和删除最大最小值相似。难点在于左右子树都不为空的情况,将当前节点设置为右子树的最小节点,挂载到当前节点的位置。
java
public class BST<E extends Comparable<E>> {
public void remove(E e) {
remove(root, e);
}
private Node remove(Node node, E e) {
if (node == null) {
return null;
}
if (e.compareTo(node.e) < 0) {
// 如果 e<node.e 则当前节点去左子树去找
node.left = remove(node.left, e);
return node;
} else if (e.compareTo(node.e) > 0) {
// 如果 e>node.e 则当前节点去右子树去找
node.right = remove(node.right, e);
return node;
} else {
// 相等的情况,开始删除逻辑
if (node.left == null) {
// 左子树为空,将右子树挂载到当前节点,删除当前节点
Node rightNode = node.right;
node.right = null;
size--;
// 理解递归逻辑,直接返回右子树相当于完成了挂载逻辑
return rightNode;
}
if (node.right == null) {
// 右子树为空,将左子树挂载到当前节点,删除当前节点
Node leftNode = node.left;
node.left = null;
size--;
return leftNode;
}
// 左右子树都不为空,将当前节点设置为右子树的最小节点,挂载到当前节点的位置
Node successor = minimum(node.right); // 获取右子树最小值
// 以successor为根节点,右子树就是删除右子树最小值后的node节点
successor.right = removeMin(node.right);
// 以successor为根节点,左子树是不变的
successor.left = node.left;
// 原节点置空,交给GC处理
node.left = node.right = null;
return successor;
}
}
}
扩展:非递归实现前序遍历
非递归需要借助”栈“数据结构,详细代码如下:
java
public void preOrderNT() {
Stack<Node> stack = new Stack<>();
stack.push(root);
while (!stack.isEmpty()) {
Node cur = stack.pop();
System.out.println(cur.e);
if (cur.right != null)
stack.push(cur.right);
if (cur.left != null)
stack.push(cur.left);
}
}
扩展:二分搜索树的层序遍历
层序遍历,顾名思义就是一层层的遍历,借助”队列“数据结构实现。
java
public void levelOrder() {
Queue<Node> queue = new LinkedList<>();
queue.add(root);
while (!queue.isEmpty()) {
Node cur = queue.remove();
System.out.println(cur.e);
if (cur.left != null) {
queue.add(cur.left);
}
if (cur.right != null) {
queue.add(cur.right);
}
}
}
TIP
思考:根据层序遍历实现加强版toStringPlus方法,采用树状形式显示,难点在于空格的显示。