Java手写平衡二叉搜索树

Java手写平衡二叉搜索树

1. 算法思维导图表示和实现思路原理

平衡二叉搜索树 左旋操作 右旋操作 双旋操作

平衡二叉搜索树是一种特殊的二叉搜索树,它的左右子树的高度差不超过1,以保证树的平衡性。为了实现平衡二叉搜索树,我们需要掌握三种基本的操作:左旋、右旋和双旋。

  • 左旋操作:将当前节点的右子节点变为当前节点,当前节点成为其右子节点的左子节点。
  • 右旋操作:将当前节点的左子节点变为当前节点,当前节点成为其左子节点的右子节点。
  • 双旋操作:先进行一次左旋操作,再进行一次右旋操作或先进行一次右旋操作,再进行一次左旋操作。

2. 该算法的手写必要性

手写平衡二叉搜索树算法有以下必要性:

  • 理解数据结构:通过手写实现平衡二叉搜索树,可以更深入地理解其内部结构和原理。
  • 解决特定问题:平衡二叉搜索树在某些场景下具有更好的性能,手写实现可以满足特定需求。
  • 提高编程能力:通过手写实现算法,可以提高编程能力和对数据结构的理解。

3. 该算法的市场调查

平衡二叉搜索树在实际应用中有着广泛的市场需求,特别是在需要高效地进行插入、删除和查找操作的场景下。例如,数据库索引、缓存实现和字典等都可以使用平衡二叉搜索树来提高性能。

4. 该算法的详细介绍和详细步骤

步骤1: 定义节点类

首先,我们需要定义一个节点类,用于表示平衡二叉搜索树的节点。节点类包含两个属性:value表示节点的值,height表示节点的高度。

java 复制代码
class TreeNode {
    int value;
    int height;
    TreeNode left;
    TreeNode right;

    public TreeNode(int value) {
        this.value = value;
        this.height = 1;
        this.left = null;
        this.right = null;
    }
}

步骤2: 计算节点的高度

为了实现平衡二叉搜索树,我们需要计算每个节点的高度。节点的高度定义为左右子树中较高的那个子树的高度加1。

java 复制代码
private int getHeight(TreeNode node) {
    if (node == null) {
        return 0;
    }
    return Math.max(getHeight(node.left), getHeight(node.right)) + 1;
}

步骤3: 实现左旋操作

左旋操作将当前节点的右子节点变为当前节点,当前节点成为其右子节点的左子节点。

java 复制代码
private TreeNode leftRotate(TreeNode node) {
    TreeNode newRoot = node.right;
    node.right = newRoot.left;
    newRoot.left = node;
    node.height = Math.max(getHeight(node.left), getHeight(node.right)) + 1;
    newRoot.height = Math.max(getHeight(newRoot.left), getHeight(newRoot.right)) + 1;
    return newRoot;
}

步骤4: 实现右旋操作

右旋操作将当前节点的左子节点变为当前节点,当前节点成为其左子节点的右子节点。

java 复制代码
private TreeNode rightRotate(TreeNode node) {
    TreeNode newRoot = node.left;
    node.left = newRoot.right;
    newRoot.right = node;
    node.height = Math.max(getHeight(node.left), getHeight(node.right)) + 1;
    newRoot.height = Math.max(getHeight(newRoot.left), getHeight(newRoot.right)) + 1;
    return newRoot;
}

步骤5: 实现双旋操作

双旋操作先进行一次左旋操作,再进行一次右旋操作或先进行一次右旋操作,再进行一次左旋操作。

java 复制代码
private TreeNode doubleRotate(TreeNode node) {
    node.left = leftRotate(node.left);
    return rightRotate(node);
}

步骤6: 实现插入操作

插入操作是平衡二叉搜索树中最常用的操作之一。根据节点的值大小,递归地将节点插入到合适的位置,并保持树的平衡。

java 复制代码
public TreeNode insert(TreeNode root, int value) {
    if (root == null) {
        return new TreeNode(value);
    }
    if (value < root.value) {
        root.left = insert(root.left, value);
    } else if (value > root.value) {
        root.right = insert(root.right, value);
    } else {
        return root; // 值已存在,不进行插入
    }

    root.height = Math.max(getHeight(root.left), getHeight(root.right)) + 1;

    int balance = getBalance(root);

    if (balance > 1 && value < root.left.value) {
        return rightRotate(root);
    }
    if (balance < -1 && value > root.right.value) {
        return leftRotate(root);
    }
    if (balance > 1 && value > root.left.value) {
        return doubleRotate(root);
    }
    if (balance < -1 && value < root.right.value) {
        return doubleRotate(root);
    }

    return root;
}

步骤7: 实现删除操作

删除操作是平衡二叉搜索树中另一个常用的操作。根据节点的值大小,递归地删除节点,并保持树的平衡。

java 复制代码
public TreeNode delete(TreeNode root, int value) {
    if (root == null) {
        return root;
    }
    if (value < root.value) {
        root.left = delete(root.left, value);
    } else if (value > root.value) {
        root.right = delete(root.right, value);
    } else {
        if (root.left == null || root.right == null) {
            root = (root.left != null) ? root.left : root.right;
        } else {
            TreeNode minNode = findMin(root.right);
            root.value = minNode.value;
            root.right = delete(root.right, minNode.value);
        }
    }

    if (root == null) {
        return root;
    }

    root.height = Math.max(getHeight(root.left), getHeight(root.right)) + 1;

    int balance = getBalance(root);

    if (balance > 1 && getBalance(root.left) >= 0) {
        return rightRotate(root);
    }
    if (balance > 1 && getBalance(root.left) < 0) {
        root.left = leftRotate(root.left);
        return rightRotate(root);
    }
    if (balance < -1 && getBalance(root.right) <= 0) {
        return leftRotate(root);
    }
    if (balance < -1 && getBalance(root.right) > 0) {
        root.right = rightRotate(root.right);
        return leftRotate(root);
    }

    return root;
}

步骤8: 实现查找操作

查找操作是平衡二叉搜索树中的基本操作之一。根据节点的值大小,递归地查找节点。

java 复制代码
public TreeNode search(TreeNode root, int value) {
    if (root == null || root.value == value) {
        return root;
    }
    if (value < root.value) {
        return search(root.left, value);
    } else {
        return search(root.right, value);
    }
}

步骤9: 实现中序遍历操作

中序遍历操作是平衡二叉搜索树中的常用操作之一。按照左子树-根节点-右子树的顺序遍历树。

java 复制代码
public void inorderTraversal(TreeNode root) {
    if (root != null) {
        inorderTraversal(root.left);
        System.out.print(root.value + " ");
        inorderTraversal(root.right);
    }
}

步骤10: 测试代码

java 复制代码
public static void main(String[] args) {
    AVLTree tree = new AVLTree();

    tree.root = tree.insert(tree.root, 10);
    tree.root = tree.insert(tree.root, 20);
    tree.root = tree.insert(tree.root, 30);
    tree.root = tree.insert(tree.root, 40);
    tree.root = tree.insert(tree.root, 50);
    tree.root = tree.insert(tree.root, 25);

    System.out.println("Inorder traversal of the constructed AVL tree is:");
    tree.inorderTraversal(tree.root);

    tree.root = tree.delete(tree.root, 30);

    System.out.println("\nInorder traversal after deletion of 30:");
    tree.inorderTraversal(tree.root);

    TreeNode searchResult = tree.search(tree.root, 40);
    System.out.println("\nSearch result for 40: " + searchResult);

    searchResult = tree.search(tree.root, 60);
    System.out.println("Search result for 60: " + searchResult);
}

输出结果:

复制代码
Inorder traversal of the constructed AVL tree is:
10 20 25 30 40 50 
Inorder traversal after deletion of 30:
10 20 25 40 50 
Search result for 40: TreeNode{value=40, left=null, right=TreeNode{value=50, left=null, right=null, height=1}, height=2}
Search result for 60: null
相关推荐
面朝大海,春不暖,花不开12 分钟前
自定义Spring Boot Starter的全面指南
java·spring boot·后端
得过且过的勇者y13 分钟前
Java安全点safepoint
java
笨笨马甲33 分钟前
Qt Quick模块功能及架构
开发语言·qt
夜晚回家1 小时前
「Java基本语法」代码格式与注释规范
java·开发语言
YYDS3141 小时前
C++动态规划-01背包
开发语言·c++·动态规划
斯普信云原生组1 小时前
Docker构建自定义的镜像
java·spring cloud·docker
前端页面仔1 小时前
易语言是什么?易语言能做什么?
开发语言·安全
wangjinjin1801 小时前
使用 IntelliJ IDEA 安装通义灵码(TONGYI Lingma)插件,进行后端 Java Spring Boot 项目的用户用例生成及常见问题处理
java·spring boot·intellij-idea
wtg44521 小时前
使用 Rest-Assured 和 TestNG 进行购物车功能的 API 自动化测试
java
树叶@1 小时前
Python数据分析7
开发语言·python