基础二叉搜索树 - java - 细节狂魔

x33g5p2x  于2022-03-02 转载在 Java  
字(3.3k)|赞(0)|评价(0)|浏览(389)

概念

二叉搜索树又称二叉排序树,它或者是一棵空树,或者是具有以下性质的二叉树:
1、若它的左子树不为空,则左子树上所有节点的值都小于根结点的值。
2、若它的右子树不为空,则右子树上所有节点的值都大于根结点的值。
3、它的左右子树也分别为二叉搜索树

直接实践

准备工作:定义一个树节点的类,和二叉搜索树的类。

搜索二叉树的查找功能

假设我们已经构造好了一个这样的二叉树,如下图

我们要思考的第一个问题是如何查找某个值是否在该二叉树中?

根据上述的逻辑,我们来把搜索的方法进行完善。

搜索二叉树的插入操作。

根据上述逻辑,我们来写一个插入节点的代码。

搜索二叉树 删除节点的操作 - 难点

再来分析一下:curDummy 和 parentDummy 是怎么找到“替罪羊”的。

总程序 - 模拟实现二叉搜索树

  1. class TreeNode{
  2. public int val;
  3. public TreeNode left;
  4. public TreeNode right;
  5. public TreeNode(int val){
  6. this.val = val;
  7. }
  8. }
  9. public class BinarySearchTree {
  10. TreeNode root;
  11. //在二叉树中 寻找指定 val 值的节点
  12. // 找到了,返回其节点地址;没找到返回 null
  13. public TreeNode search(int key){
  14. TreeNode cur = this.root;
  15. while(cur != null){
  16. if(cur.val == key){
  17. return cur;
  18. }else if(cur.val < key){
  19. cur = cur.right;
  20. }else{
  21. cur = cur.left;
  22. }
  23. }
  24. return null;
  25. }
  26. // 插入操作
  27. public boolean insert(int key){
  28. if(this.root == null){
  29. this.root = new TreeNode(key);
  30. return true;
  31. }
  32. TreeNode cur = this.root;
  33. TreeNode parent = null;
  34. while(cur!=null){
  35. if(key > cur.val){
  36. parent = cur;
  37. cur = cur.right;
  38. }else if(cur.val == key){
  39. return false;
  40. }else{
  41. parent = cur;
  42. cur = cur.left;
  43. }
  44. }
  45. TreeNode node = new TreeNode(key);
  46. if(parent .val > key){
  47. parent.left = node;
  48. }else{
  49. parent.right = node;
  50. }
  51. return true;
  52. }
  53. // 删除操作
  54. public void remove(int key){
  55. TreeNode cur = root;
  56. TreeNode parent = null;
  57. // 寻找 删除节点位置。
  58. while(cur!=null){
  59. if(cur.val == key){
  60. removeNode(cur,parent);// 真正删除节点的代码
  61. break;
  62. }else if(cur.val < key){
  63. parent = cur;
  64. cur = cur.right;
  65. }else{
  66. parent = cur;
  67. cur = cur.left;
  68. }
  69. }
  70. }
  71. // 辅助删除方法:真正删除节点的代码
  72. private void removeNode(TreeNode cur,TreeNode parent){
  73. // 情况一
  74. if(cur.left == null){
  75. if(cur == this.root){
  76. this.root = this.root.right;
  77. }else if( cur == parent.left){
  78. parent.left = cur.right;
  79. }else{
  80. parent.right = cur.right;
  81. }
  82. // 情况二
  83. }else if(cur.right == null){
  84. if(cur == this.root){
  85. this.root = root.left;
  86. }else if(cur == parent.left){
  87. parent.left = cur.left;
  88. }else{
  89. parent.right = cur.left;
  90. }
  91. // 情况三
  92. }else{
  93. // 第二种方法:在删除节点的右子树中寻找最小值,
  94. TreeNode parentDummy = cur;
  95. TreeNode curDummy = cur.right;
  96. while(curDummy.left != null){
  97. parentDummy = curDummy;
  98. curDummy = curDummy.left;
  99. }
  100. // 此时 curDummy 指向的 cur 右子树
  101. cur.val = curDummy.val;
  102. if(parentDummy.left != curDummy){
  103. parentDummy.right = curDummy.right;
  104. }else{
  105. parentDummy.left = curDummy.right;
  106. }
  107. }
  108. }
  109. // 中序遍历
  110. public void inorder(TreeNode root){
  111. if(root == null){
  112. return;
  113. }
  114. inorder(root.left);
  115. System.out.print(root.val+" ");
  116. inorder(root.right);
  117. }
  118. public static void main(String[] args) {
  119. int[] array = {10,8,19,3,9,4,7};
  120. BinarySearchTree binarySearchTree = new BinarySearchTree();
  121. for (int i = 0; i < array.length; i++) {
  122. binarySearchTree.insert(array[i]);
  123. }
  124. binarySearchTree.inorder(binarySearchTree.root);
  125. System.out.println();// 换行
  126. System.out.print("插入重复的数据 9:" + binarySearchTree.insert(9));
  127. System.out.println();// 换行
  128. System.out.print("插入不重复的数据 1:" + binarySearchTree.insert(1));
  129. System.out.println();// 换行
  130. binarySearchTree.inorder(binarySearchTree.root);
  131. System.out.println();// 换行
  132. binarySearchTree.remove(19);
  133. System.out.print("删除元素 19 :");
  134. binarySearchTree.inorder(binarySearchTree.root);
  135. System.out.println();// 换行
  136. System.out.print("查找不存在的数据50 :");
  137. System.out.println(binarySearchTree.search(50));
  138. System.out.print("查找存在的数据 7:");
  139. System.out.println(binarySearchTree.search(7));
  140. }
  141. }

性能分析

插入和删除操作都必须先查找,查找效率代表了二叉搜索树中各个操作的性能。

  对有n个结点的二叉搜索树,若每个元素查找的概率相等,则二叉搜索树平均查找长度是结点在二叉搜索树的深度的函数,即结点越深,则比较次数越多。

  但对于同一个关键码集合,如果各关键码插入的次序不同,可能得到不同结构的二叉搜索树:

如果我们能保证 二叉搜索树的左右子树高度差不超过1。尽量满足高度平衡条件。
这就成 AVL 树了(高度平衡的二叉搜索树)。而AVL树,也有缺点:需要一个频繁的旋转。浪费很多效率。
至此 红黑树就诞生了,避免更多的旋转。

和 java 类集的关系

TreeMap 和 TreeSet 即 java 中利用搜索树实现的 Map 和 Set;实际上用的是红黑树,而红黑树是一棵近似平衡的二叉搜索树,即在二叉搜索树的基础之上 + 颜色以及红黑树性质验证,关于红黑树的内容,等博主学了,会写博客的。

相关文章

最新文章

更多