This is the 13th day of my participation in the November Gwen Challenge. Check out the event details: The last Gwen Challenge 2021

The statement

This article is the basic introduction of binary search tree (also called sorting binary tree), and related operations, and finally with code description, this article mainly uses the programming language for Java.

define

A binary search tree, literally, is essentially a specially structured binary tree, according to Wikipedia: a binary tree with a root, a key (and an optional association value) stored inside each node, and two distinct subtrees, usually represented as left and right.

This article mainly uses Java implementation, each node will store a value, there is no so-called key.

According to the properties of binary search tree, when the root node of a binary search tree is not empty, the node distribution should comply with the following rules:

  1. The left subtree is non-empty, and the values of all nodes in the left subtree are less than the values of the parent node of the left subtree
  2. The right subtree is non-empty, and the value of all nodes in the right subtree is greater than the value of the parent node of the right subtree
  3. The left and right subtree should also be a binary sort tree

Binary search trees are used to construct more abstract data structures, such as sets and associative arrays. (This is Wiki saying, not me)

operation

The operations supported by binary search trees include insert, delete, and query.

When performing these operations on a binary search tree, the time complexity of insert, delete, and query operations is related to the depth of the tree (similar to binary search). For example, if the number of nodes in the tree is N, the time complexity is O(log n), where n is the number of nodes in the tree.

When the time complexity is O(n) in the worst case, the tree degenerates into a linked list. For details, see 2.3 Node Query Operation Introduction.

insert

According to the nature of the binary search tree, the insertion operation needs to meet the following conditions (i.e. the insertion step) :

  1. If the tree is empty (the root node points to empty), the new node is the root node, otherwise the root node is used as the current node to start the search
  2. If the value of the newly inserted node is smaller than that of the node being compared and the left subtree of the node being compared is not empty, the search continues with the left child of the node being compared as the current node
  3. If the value of the newly inserted node is not less than that of the node being compared and the right subtree of the node being compared is not empty, the search continues with the right child node of the node being compared as the current node
  4. Repeat steps 2 or 3 until you find a node that satisfies condition 5
  5. If the new node is smaller than the value of the current node and the left subtree of the current node is empty, the new node serves as the left child of the current node. Otherwise, if the new node is not smaller than the current node and the right subtree of the current node is empty, the new node serves as the right child of the current node.

Take the example of inserting an array [18,5,4,66,32,78], as shown in the figure below.

If an intermediate traversal is performed, it is arranged in the smallest order: 4,5,18,32,66,78

delete

After deleting a node of the binary search tree, ensure that the deleted binary search tree is still a binary search tree structure. When deleting the binary search tree, follow the following rules (press Method/Skill to delete the node) :

  1. The node to be deleted is the root node and there is only one root node. Delete the root node (the root node points to empty).
  2. The node to be deleted is a leaf node and can be removed from the parent node
  3. The deleted node has only the left child node, and the left child node is the left child node of its parent node
  4. The deleted node has only the right child node, and the right child node is the right child node of its parent node
  5. Deleted nodes around the child nodes are not empty, find predecessor deleted node (before subsequent nodes to also go, this is, for example), the former is the largest of the left subtree is node (which is most the right side of the node in the left subtree, according to the nature, the former node or a child node, or no leaf nodes), Repeat steps 2,3, and 4 until they are deleted. Of course, there are other methods to delete, the reader can refer to other information to understand, this paper to explain the method.

Cases 1, 2, 3, 4 and 5 can be understood literally. The key legend illustrates the deletion operation of the fifth case:

The middle-order traversal operations for node 66 are as follows: 4,5,18,32,78

The query

In fact, the query operation is to compare the value of the queried node with one of the nodes at each level of the tree until the node with the same value is found, or at last return null. Insert and delete also apply query operations.

Because each layer is compared with only one node (either the left child or the right child), its time complexity is O(log n), depending on the depth of the tree, but in the worst case, that is, the binary search tree degenerated into a linked list, the values inserted are ordered, such as (1,2,3,4,5). At this point, The time complexity is O(n), and the structure of the tree is shown as follows:

Java source code implementation

package com.xuxd.binary.search;

import java.util.ArrayList;
import java.util.List;

/ * * *@DescriptionBinary search tree Java implementation */
public class BinarySearchTree<E extends Comparable> {

    /** * Root node */
    private Node root;

    public BinarySearchTree(a) {}public BinarySearchTree(E e) {
        this.root = new Node(e, null.null.null);
    }

    /** * add a node **@paramThe value of e node */
    public void add(E e) {
        // The root node is empty
        if (root == null) {
            root = new Node(e, null.null.null);
        } else {
            Node current = root;
            Node parent;
            boolean lessThanLeft;
            do {
                parent = current;
                lessThanLeft = e.compareTo(current.data) < 0 ? true : false;
                // If the new node is smaller than the left child of the current node, the search continues with the left child of the current node
                if (lessThanLeft) {
                    current = parent.left;
                } else {
                    If the new node is not smaller than the left child of the current node, the search continues with the right child of the current node as the current nodecurrent = parent.right; }}while(current ! =null);
            Node newNode = new Node(e, parent, null.null);
            if (lessThanLeft) {
                parent.left = newNode;
            } else{ parent.right = newNode; }}}/** * Deletes the specified node ** based on the entered value@paramE The value of the node to be deleted */
    public void remove(E e) {
        // Get the node to delete
        Node delNode = getNode(e);
        while(delNode ! =null) {
            // If the node to be deleted is a leaf node, remove it directly
            if (delNode.left == null && delNode.right == null) {
                if (delNode == root) {
                    root = null;
                } else {
                    If the node to be removed is the left child of its parent, remove the left child of its parent, otherwise remove the right child of its parent
                    if (delNode == delNode.parent.left) {
                        delNode.parent.left = null;
                    } else {
                        delNode.parent.right = null;
                    }
                    delNode.parent = null;
                }
                delNode = null;
            } else if(delNode.left ! =null && delNode.right == null) {
                // The deleted node has only the left child node
                if (delNode == root) {
                    // If it is the root node, make the left child the root node
                    root = delNode.left;
                    root.parent = null;
                    delNode = null;
                } else {
                    delNode.left.parent = delNode.parent;
                    delNode.parent.left = delNode.left;
                    delNode = null; }}else if (delNode.left == null&& delNode.right ! =null) {
                // The deleted node has only the right child node
                if (delNode == root) {
                    // If it is the root node, make the right child the root node
                    root = delNode.right;
                    root.parent = null;
                    delNode.right = null;
                    delNode = null;
                } else {
                    delNode.right.parent = delNode.parent;
                    delNode.parent.right = delNode.right;
                    delNode = null; }}else {
                // Both the left and right child nodes of the deleted node exist
                // Find the largest node in the left subtree of the deleted node, which is the rightmost node in the left subtree
                Node leftMaxNode = delNode.left;
                while(leftMaxNode.right ! =null) { leftMaxNode = leftMaxNode.right; } delNode.data = leftMaxNode.data; delNode = leftMaxNode; }}}/** * finds the node ** with the specified value@paramThe value of the e node *@returnEligible nodes or NULL */
    public Node getNode(E e) {
        Node p = root;
        int cmp;
        while(p ! =null) {
            cmp = e.compareTo(p.data);
            if (cmp < 0) {
                p = p.left;
            } else if (cmp > 0) {
                p = p.right;
            } else {
                returnp; }}return null;
    }

    // Prints the value of each node in the tree in middle order traversal
    public void print(a) {
        List<Node> nodes = new ArrayList<Node>();
        if(root ! =null) {
            inorderTraversal(root, nodes);
        }
        System.out.println(nodes);
    }

    // middle order traversal
    private List<Node> inorderTraversal(Node node, List<Node> nodes) {
        if(node.left ! =null) {
            inorderTraversal(node.left, nodes);
        }
        nodes.add(node);
        if(node.right ! =null) {
            inorderTraversal(node.right, nodes);
        }
        return nodes;
    }

    /** * Node class */
    private static class Node {
        private Object data;
        private Node parent;
        private Node left;
        private Node right;

        public Node(Object data) {
            this.data = data;
        }

        public Node(Object data, Node parent, Node left, Node right) {
            this.data = data;
            this.parent = parent;
            this.left = left;
            this.right = right;
        }

        @Override
        public boolean equals(Object obj) {
            if (this == obj) {
                return true;
            }
            if (this.getClass() == obj.getClass()) {
                Node target = (Node) obj;
                return data.equals(target.data) && parent == target.parent && left == target.left && right == target.right;
            }
            return false;
        }

        @Override
        public String toString(a) {
            returndata.toString(); }}public static void main(String[] args) {
        BinarySearchTree<Integer> tree = new BinarySearchTree<Integer>();
        Integer[] integers = {18.5.4.66.32.78};
        for (Integer i :
                integers) {
            tree.add(i);
        }
        tree.print();
        for(Integer i : integers) { tree.remove(i); tree.print(); }}}Copy the code

References:

Binary search tree – Wiki

Crazy Java: 16 Lessons to Break Basic Programmer Skills