当前位置:首页 > 编程笔记 > 正文
已解决

Java手写平衡二叉搜索树

来自网友在路上 171871提问 提问时间:2023-09-19 01:56:10阅读次数: 71

最佳答案 问答题库718位专家为你答疑解惑

Java手写平衡二叉搜索树

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

<style>#mermaid-svg-KcRS9eQKGtO9QHsv {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-KcRS9eQKGtO9QHsv .error-icon{fill:#552222;}#mermaid-svg-KcRS9eQKGtO9QHsv .error-text{fill:#552222;stroke:#552222;}#mermaid-svg-KcRS9eQKGtO9QHsv .edge-thickness-normal{stroke-width:2px;}#mermaid-svg-KcRS9eQKGtO9QHsv .edge-thickness-thick{stroke-width:3.5px;}#mermaid-svg-KcRS9eQKGtO9QHsv .edge-pattern-solid{stroke-dasharray:0;}#mermaid-svg-KcRS9eQKGtO9QHsv .edge-pattern-dashed{stroke-dasharray:3;}#mermaid-svg-KcRS9eQKGtO9QHsv .edge-pattern-dotted{stroke-dasharray:2;}#mermaid-svg-KcRS9eQKGtO9QHsv .marker{fill:#333333;stroke:#333333;}#mermaid-svg-KcRS9eQKGtO9QHsv .marker.cross{stroke:#333333;}#mermaid-svg-KcRS9eQKGtO9QHsv svg{font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;}#mermaid-svg-KcRS9eQKGtO9QHsv .label{font-family:"trebuchet ms",verdana,arial,sans-serif;color:#333;}#mermaid-svg-KcRS9eQKGtO9QHsv .cluster-label text{fill:#333;}#mermaid-svg-KcRS9eQKGtO9QHsv .cluster-label span{color:#333;}#mermaid-svg-KcRS9eQKGtO9QHsv .label text,#mermaid-svg-KcRS9eQKGtO9QHsv span{fill:#333;color:#333;}#mermaid-svg-KcRS9eQKGtO9QHsv .node rect,#mermaid-svg-KcRS9eQKGtO9QHsv .node circle,#mermaid-svg-KcRS9eQKGtO9QHsv .node ellipse,#mermaid-svg-KcRS9eQKGtO9QHsv .node polygon,#mermaid-svg-KcRS9eQKGtO9QHsv .node path{fill:#ECECFF;stroke:#9370DB;stroke-width:1px;}#mermaid-svg-KcRS9eQKGtO9QHsv .node .label{text-align:center;}#mermaid-svg-KcRS9eQKGtO9QHsv .node.clickable{cursor:pointer;}#mermaid-svg-KcRS9eQKGtO9QHsv .arrowheadPath{fill:#333333;}#mermaid-svg-KcRS9eQKGtO9QHsv .edgePath .path{stroke:#333333;stroke-width:2.0px;}#mermaid-svg-KcRS9eQKGtO9QHsv .flowchart-link{stroke:#333333;fill:none;}#mermaid-svg-KcRS9eQKGtO9QHsv .edgeLabel{background-color:#e8e8e8;text-align:center;}#mermaid-svg-KcRS9eQKGtO9QHsv .edgeLabel rect{opacity:0.5;background-color:#e8e8e8;fill:#e8e8e8;}#mermaid-svg-KcRS9eQKGtO9QHsv .cluster rect{fill:#ffffde;stroke:#aaaa33;stroke-width:1px;}#mermaid-svg-KcRS9eQKGtO9QHsv .cluster text{fill:#333;}#mermaid-svg-KcRS9eQKGtO9QHsv .cluster span{color:#333;}#mermaid-svg-KcRS9eQKGtO9QHsv div.mermaidTooltip{position:absolute;text-align:center;max-width:200px;padding:2px;font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:12px;background:hsl(80, 100%, 96.2745098039%);border:1px solid #aaaa33;border-radius:2px;pointer-events:none;z-index:100;}#mermaid-svg-KcRS9eQKGtO9QHsv :root{--mermaid-font-family:"trebuchet ms",verdana,arial,sans-serif;}</style>
平衡二叉搜索树
左旋操作
右旋操作
双旋操作

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

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

2. 该算法的手写必要性

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

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

3. 该算法的市场调查

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

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

步骤1: 定义节点类

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

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。

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

步骤3: 实现左旋操作

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

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: 实现右旋操作

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

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: 实现双旋操作

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

private TreeNode doubleRotate(TreeNode node) {node.left = leftRotate(node.left);return rightRotate(node);
}

步骤6: 实现插入操作

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

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: 实现删除操作

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

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: 实现查找操作

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

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: 实现中序遍历操作

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

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

步骤10: 测试代码

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
查看全文

99%的人还看了

猜你感兴趣

版权申明

本文"Java手写平衡二叉搜索树":http://eshow365.cn/6-9038-0.html 内容来自互联网,请自行判断内容的正确性。如有侵权请联系我们,立即删除!