数据结构----二叉树之二叉搜索树

文章目录

1. 概述

  1. 二叉树规则
    1. 只能存在一个根节点。
    2. 每个节点最多只能存在两个子节点。
    3. 每个节点最多只能存在一个父亲节点。
  2. 二叉搜索树规则
    1. 拥有二叉树的规则。
    2. 大于其左子树所有节点的值。
    3. 小于其右子树所有节点的值。
    4. 存储的元素必须具有“可比较性”。
    5. 使用中序遍历,可以得到升序排列的数据。

2. 源码

/**
 * 二叉搜索树
 * 时间复杂度是O(log n)
 */
public class BinarySearchTree<T extends Comparable<T>> {

    private class Node {
        private T data;

        private Node left, right;

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

    private Node root;

    private int size;

    public int getSize() {
        return size;
    }

    public boolean isEmpty() {
        return size == 0;
    }

    public void add(T data) {
        root = add(root, data);
    }

    // 递归插入节点.
    private Node add(Node node, T data) {
        if (node == null) {
            size++;
            return new Node(data);
        }

        int compare = data.compareTo(node.data);
        if (compare > 0) {
            node.right = add(node.right, data);
        } else if (compare < 0) {
            node.left = add(node.left, data);
        }

        return node;
    }

    // 查询节点.
    public boolean contains(T data) {
        return contains(root, data);
    }

    private boolean contains(Node node, T data) {
        if (node == null) {
            return false;
        }

        int compare = data.compareTo(node.data);
        if (compare == 0) {
            return true;
        } else if (compare > 0) {
            return contains(node.right, data);
        } else {
            return contains(node.left, data);
        }
    }

    // 前序遍历
    public void preOrder() {
        preOrder(root);
    }

    // 递归进行前序遍历
    private void preOrder(Node node) {
        if (node == null) {
            return;
        }

        System.out.println(node.data);
        preOrder(node.left);
        preOrder(node.right);
    }

    // 前序遍历非递归写法.
    public void preOrderNR() {
        Stack<Node> stack = new Stack<>();
        stack.push(root);
        while (!stack.isEmpty()) {
            Node cur = stack.pop();
            System.out.println(cur.data);

            // 必须先放right, 再放left, 保证下一次stack弹出的是left.
            if (cur.right != null) {
                stack.push(cur.right);
            }

            if (cur.left != null) {
                stack.push(cur.left);
            }
        }
    }

    // 层序遍历非递归写法(广度优先).
    public void levelOrder() {
        LinkedList<Node> list = new LinkedList<>();
        list.addLast(root);
        while (!list.isEmpty()) {
            Node cur = list.removeFirst();
            System.out.println(cur.data);

            if (cur.left != null) {
                list.addLast(cur.left);
            }

            if (cur.right != null) {
                list.addLast(cur.right);
            }
        }
    }

    // 中序遍历: 遍历出来的元素顺序是升序.
    public void inOrder() {
        inOrder(root);
    }

    // 递归进行中序遍历
    private void inOrder(Node node) {
        if (node == null) {
            return;
        }

        inOrder(node.left);
        System.out.println(node.data);
        inOrder(node.right);
    }

    // 递归进行后序遍历
    public void postOrder() {
        postOrder(root);
    }

    private void postOrder(Node node) {
        if (node == null) {
            return;
        }

        postOrder(node.left);
        postOrder(node.right);
        System.out.println(node.data);
    }

    // 获取最小节点数据(递归).
    public T getMinimum() {
        if (isEmpty()) {
            throw new IllegalArgumentException("The tree is empty!");
        }

        return getMinimum(root).data;
    }

    private Node getMinimum(Node node) {
        if (node.left == null) {
            return node;
        }

        return getMinimum(node.left);
    }

    // 获取最大节点数据(递归).
    public T getMaxmum() {
        if (isEmpty()) {
            throw new IllegalArgumentException("The tree is empty!");
        }

        return getMaxmum(root).data;
    }

    private Node getMaxmum(Node node) {
        if (node.right == null) {
            return node;
        }

        return getMaxmum(node.right);
    }

    // 删除最小值所在的节点, 返回最小值.(递归)
    public T removeMin() {
        T result = getMinimum();
        // 需要从树中删除此节点
        root = removeMin(root);
        return result;
    }

    // 删除以node为根的二分搜索树中的最小节点
    // 返回删除节点后新的二分搜索树的根
    private Node removeMin(Node node) {
        if (node.left == null) {
            Node rightNode = node.right;
            node.right = null;
            size--;
            return rightNode;
        }

        node.left = removeMin(node.left);
        return node;
    }

    // 删除最大值所在的节点, 返回最大值.(递归)
    public T removeMax() {
        T result = getMaxmum();
        // 需要从树中删除此节点
        root = removeMax(root);
        return result;
    }

    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;
    }

    // 删除节点
    public void remove(T data) {
        root = remove(root, data);
    }

    // 递归删除节点
    private Node remove(Node node, T data) {
        if (node == null) {
            return null;
        }

        int compare = data.compareTo(node.data);
        if (compare > 0) {
            node.right = remove(node.right, data);
            return node;
        } else if (compare < 0) {
            node.left = remove(node.left, data);
            return node;
        } else { // data == node.data
            // 待删除的节点左子树为空的情况
            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 = getMinimum(node.right);
            successor.right = removeMin(node.right);
            successor.left = node.left;

            node.left = node.right = null;
            return successor;
        }
    }

    @Override
    public String toString() {
        StringBuilder result = new StringBuilder();
        generatorToString(root, 0, result);
        return result.toString();
    }

    // 生成以node为根节点,深度为depth的描述二叉树的字符串
    private void generatorToString(Node node, int depth, StringBuilder result) {
        if (node == null) {
            result.append(generatorDepthString(depth) + "null\n");
            return;
        }

        result.append(generatorDepthString(depth) + node.data + "\n");
        generatorToString(node.left, depth + 1, result);
        generatorToString(node.right, depth + 1, result);
    }

    private String generatorDepthString(int depth) {
        StringBuilder result = new StringBuilder();
        for (int i = 0; i < depth; i++) {
            result.append("--");
        }

        return result.toString();
    }

    public static void main(String[] args) {
        BinarySearchTree<Integer> tree = new BinarySearchTree<>();
        int[] nums = {10, 13, 15, 8, 7, 9};

        for (int num : nums) {
            tree.add(num);
        }


        System.out.println("==========Pre Order==============");
        tree.preOrder();
        System.out.println("==========Pre OrderNR==============");
        tree.preOrderNR();
        System.out.println("==========In Order==============");
        tree.inOrder();
        System.out.println("==========After Order==============");
        tree.postOrder();
        System.out.println("==========Pre OrderNR2==============");
        tree.levelOrder();
        System.out.println("==========Get minimum Tree Node==============");
        System.out.println(tree.getMinimum());
        System.out.println("==========Remove minimum Tree Node Before==============");
        System.out.println(tree.removeMin());
        System.out.println("==========Remove minimum Tree Node After==============");
        tree.inOrder();
        System.out.println("==========Remove maxmum Tree Node Before==============");
        System.out.println(tree.removeMax());
        System.out.println("==========Remove maxmum Tree Node After==============");
        tree.inOrder();
    }
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值