一、概念介绍

1. binary search(sort) tree

2. 现实逻辑

image.png

3. 代码实现

target: 对一个数组进行二叉排序树排列,并按照中序遍历写出结果

3.1 创建Node类

  1. class Node{
  2. int value;
  3. Node left;
  4. Node right;
  5. public Node(int value) {
  6. this.value = value;
  7. }
  8. // 添加节点方法
  9. // 递归的方式添加节点,注意满足二叉排序树的要求
  10. public void add(Node node){
  11. if(node == null){
  12. return;
  13. }
  14. // 判断传入的节点的值和当前子树根节点的关系
  15. if(node.value<this.value){
  16. // 左子节点为空和不为空2种情况
  17. if(this.left==null){
  18. this.left=node;
  19. }else {
  20. this.left.add(node);
  21. }
  22. }else {
  23. if(this.right==null){
  24. this.right=node;
  25. }else {
  26. this.right.add(node);
  27. }
  28. }
  29. }
  30. // 中序遍历二叉树
  31. public void infixOrder(){
  32. // left
  33. if(this.left!=null){
  34. this.left.infixOrder();
  35. }
  36. // middle
  37. System.out.println(this.value);
  38. // right
  39. if(this.right!=null){
  40. this.right.infixOrder();
  41. }
  42. }
  43. }

3.2 创建二叉排序树

  1. class BinarySortTree{
  2. private Node root;
  3. public void add(Node node){
  4. if(root==null){// empty tree
  5. root=node;
  6. }else{
  7. root.add(node);
  8. }
  9. }
  10. // infixOrder
  11. public void infixOrder(){
  12. if(root!=null){
  13. root.infixOrder();
  14. }else {
  15. System.out.println("current binaryTree is empty!");
  16. }
  17. }
  18. }

3.3 传入数组,现实功能

  1. public class BinarySortTreeDemo {
  2. public static void main(String[] args) {
  3. int[] arr = {7, 3, 10, 12, 5, 1, 9};
  4. BinarySortTree binarySortTree = new BinarySortTree();
  5. for (int i = 0; i < arr.length; i++) {
  6. binarySortTree.add(new Node(arr[i]));
  7. }
  8. binarySortTree.infixOrder();
  9. }
  10. }

3.4 结果

image.png

二、二叉排序树的删除

1. 情况分析

image.png

2. 逻辑思维

image.png

  1. 第一种情况:
  2. 删除叶子节点 (比如:2, 5, 9, 12)
  3. 思路
  4. (1) 需求先去找到要删除的结点 targetNode
  5. (2) 找到targetNode 父结点 parent
  6. (3) 确定 targetNode parent的左子结点 还是右子结点
  7. (4) 根据前面的情况来对应删除
  8. 左子结点 parent.left = null
  9. 右子结点 parent.right = null;
  1. 第二种情况: 删除只有一颗子树的节点 比如 1
  2. 思路
  3. (1) 需求先去找到要删除的结点 targetNode
  4. (2) 找到targetNode 父结点 parent
  5. (3) 确定targetNode 的子结点是左子结点还是右子结点
  6. (4) targetNode parent 的左子结点还是右子结点
  7. (5) 如果targetNode 有左子结点
  8. 5. 1 如果 targetNode parent 的左子结点
  9. parent.left = targetNode.left;
  10. 5.2 如果 targetNode parent 的右子结点
  11. parent.right = targetNode.left;
  12. (6) 如果targetNode 有右子结点
  13. 6.1 如果 targetNode parent 的左子结点
  14. parent.left = targetNode.right;
  15. 6.2 如果 targetNode parent 的右子结点
  16. parent.right = targetNode.right
  1. 情况三 删除有两颗子树的节点. (比如:7, 310 )
  2. 思路
  3. (1) 需求先去找到要删除的结点 targetNode
  4. (2) 找到targetNode 父结点 parent
  5. (3) targetNode 的右子树找到最小的结点
  6. (4) 用一个临时变量,将 最小结点的值保存 temp = 11
  7. (5) 删除该最小结点
  8. (6) targetNode.value = temp

3. 代码实现

3.1 Node类中查找目标节点

// 查找要删除的节点
    public Node search(int value) {
        if (value == this.value) {
            return this;
        } else if (value < this.value) {//向左递归
            // 如果左子节点为空,则停止查找
            if (this.left == null) {
                return null;
            }
            return this.left.search(value);

        } else {//向左递归
            // 如果左子节点为空,则停止查找
            if (this.right == null) {
                return null;
            }
            return this.right.search(value);
        }
    }

3.2 Node类中查找目标节点的父节点

/**
     * 查找要删除节点的父节点
     * @param value 目标节点的值
     * @return 父节点
     */
    public Node searchParent(int value){
        if((this.left!=null&& this.left.value==value)||(this.right!=null&& this.right.value==value)){
            return this;
        }else {
            // 如果查找的节点比当前节点小,并且当前节点左子节点不为空
            if(value<this.value && this.left!=null){
                return this.left.search(value); //向左递归查找
            }else if(value>this.value && this.right!=null){
                return this.right.search(value);
            }else {
                return null;
            }
        }
    }

3.3 BinarySearchTree中

// 查找要删除的节点
    public Node searchParent(int value){
        if(root==null){
            return null;
        }else {
            return root.searchParent(value);
        }
    }
//删除节点
    public void delNode(int value){
        if(root==null){
            return;
        }else {
            // 1. 找到target节点
            Node target =search(value);
            // 如果没有找到要删除的节点
            if(target==null){
                return;
            }
            // 如果我们发现当前这颗二叉排序树只有一个节点
            if(root.left==null && root.right==null){
                root=null;
                return;
            }
            // 找到target父节点
            Node parentNode = searchParent(value);
            // 如果要删除的节点是叶子节点
            if(target.left==null && target.right==null){
                // 判断target是父节点的左子节点还是右子节点
                if(parentNode.left!=null && parentNode.left.value==value){
                    parentNode.left=null;
                }else if(parentNode.right!=null && parentNode.right.value==value){
                    parentNode.right=null;
                }
            }
        }
    }