AVL树

在计算机科学中,AVL树是最先发明的自平衡二叉查找树。在AVL树中任何节点的两个子树的高度最大差别为1,所以它也被称为高度平衡树。增加和删除可能需要通过一次或多次树旋转来重新平衡这个树。AVL树得名于它的发明者G. M. Adelson-Velsky和E. M. Landis,他们在1962年的论文《An algorithm for the organization of information》中发表了它。

平衡因子:某结点的左子树与右子树的高度(深度)差即为该结点的平衡因子

AVL树的特点

  1. 本身首先是一棵二叉搜索树。

  2. 带有平衡条件:每个结点的左右子树的高度之差的绝对值(平衡因子)最多为1。

    也就是说,AVL树,本质上是带了平衡功能的二叉查找树(二叉排序树,二叉搜索树)

为什么需要平衡树?

举个极端的例子:当我们将一个数组 [5, 6, 7, 8, 9]转换成二叉排序树时,它的二叉树图会如下图所示。

可以看出这时二叉树趋近于链表,虽然它的插入和删除的效率不会受到影响,但是查找的效率就有所降低了,因为它和链表一样需要从头到尾查找。

AVL树的旋转操作

AVL树的旋转操作有三种:

  • 当右子树的高度高于左子树一层以上时进行的左旋转

  • 当左子树的高度高于右子树一层以上时进行的右旋转

  • 以及特殊情况下进行的双向旋转

左旋转

当二叉排序树的某一节点的右子树的高度高于左子树一层以上时,为了平衡二叉树排序树,需要对二叉排序树进行左旋转操作。

右旋转

当二叉排序树的某一节点的左子树的高度高于右子树一层以上时,为了平衡二叉树排序树,需要对二叉排序树进行右旋转操作。

双向旋转

在进行左旋转或右旋转时需要判断一个特殊情况。如下图所示,根节点的左子树高度高于右子树一层以上,即平衡因子大于1,随即对该二叉树排序树进行右旋转操作,可以 看出右旋转后根节点右子树的高度高于左子树一层以上,平衡因子还是大于1,二叉排序树的平衡问题还是没解决。

观察该二叉排序树可以发现,根节点的左子节点8号节点的右子树的高度是高于左子树的,但只大于1,所以没有进入到旋转平衡操作,当我们为根节点下的子树进行右旋转时,会将8号节点较高的右子树连接到10号节点上,这时以10号节点为根节点的树高度就高于8号节点的左子树2层了,而随后我们又将该树来连接到8号节点的右子树上,就导致了旋转后平衡因子还是大于1的问题。

所以在遇到上面这种情况,即左旋转时右子树的左子树高于右子树的右子树右旋转时左子树的右子树高于左子树的左子树,需先对较高的子树树进行一次右旋转或左旋转,再对整颗树进行左旋转或右旋转。

下图是当右旋转时左子树的右子树高于左子树的左子树的情况。

代码实现

获取树的高度

在进行旋转前,需要先判断以该节点为根节点的树是否需要进行旋转平衡,而判断是否需要旋转得知道该节点的左子树和右子树的高度。

  1. //获取该节点左子树的高度
  2. public int leftHeight(){
  3. if (this.left == null){
  4. return 0;
  5. }
  6. return this.left.height();
  7. }
  8.  
  9. //获取该节点右子树的高度
  10. public int rightHeight(){
  11. if (this.right == null){
  12. return 0;
  13. }
  14. return this.right.height();
  15. }
  16.  
  17. //获取以该节点为根节点的树的高度
  18. public int height(){
  19. /*
  20. Math.max方法可以取出俩个参数中的最大值,因为是递归叠加,所以需+1,
  21. 如当该节点是叶子节点时,如果不+1的话,会返回0,导致该节点没有被算入高度中
  22. */
  23. return Math.max(this.left==null ? 0:this.left.height(), this.right==null ? 0:this.right.height()) + 1;
  24. }

旋转操作

  1. //左旋转
  2. public AVLNode leftRotate(){
  3. //保存该节点的右子树
  4. AVLNode rightSubTree = this.right;
  5. //将该节点的右子树 替换成 保存的右子树rightSubTree的左子树
  6. this.right = rightSubTree.left;
  7. //再把以该节点为根节点的树 连接到 rightSubTree的左子树
  8. rightSubTree.left = this;
  9. //最后返回左旋转后新的树
  10. return rightSubTree;
  11. }
  12.  
  13. //右旋转
  14. public AVLNode rightRotate(){
  15. AVLNode leftSubTree = this.left;
  16. this.left = leftSubTree.right;
  17. leftSubTree.right = this;
  18. return leftSubTree;
  19. }

在进行完上面步骤后,还需要进行关键的一步,把旋转后返回的新树连接到该节点在旋转前的父节点,因为本来以该节点为根节点的树经过旋转后根节点发生了变化,不再是该节点,所以需要用旋转后新树的根节点去替代旋转前该节点位置。

在进行连接父节点操作时,还需注意旋转前该节点是整颗二叉排序树中一颗子树的根节点还是整颗二叉排序树的根节点

添加操作中的平衡判断:

  1. /**
  2. * 在二叉排序树中添加节点
  3. * @param node 添加的节点
  4. * @param pointer 指针节点,用于遍历节点,初始指向根节点
  5. * @return 返回添加后平衡的新树
  6. */
  7. public AVLNode add(AVLNode node, AVLNode pointer){
  8. if (node == null){
  9. return null;
  10. }
  11.  
  12. if (pointer.value > node.value){//指针节点值大于添加节点值时
  13. //如果指针节点的左节点刚好为空,则将添加节点插入到该左节点
  14. if (pointer.left == null){
  15. pointer.left = node;
  16. }else {
  17. //如果不是则继续往左节点走
  18. pointer.left = add(node, pointer.left);
  19. }
  20. }else {//指针节点值小于添加节点值时
  21. //如果指针节点的右节点刚好为空,则将添加节点插入到该右节点
  22. if (pointer.right == null){
  23. pointer.right = node;
  24. }else {
  25. //如果不是则继续往右节点走
  26. pointer.right = add(node, pointer.right);
  27. }
  28. }
  29.  
  30. //添加完节点后,判断以该节点为根节点的树是否平衡,如不平衡则需进行旋转
  31. if (rightHeight(pointer) - leftHeight(pointer) > 1){//判断节点的右子树高度是否比左子树高过1层以上
  32. //当左旋转时,需判断右子树的左子树高度是否高于右子树的右子树
  33. if (leftHeight(pointer.right) > rightHeight(pointer.right)){
  34. //当满足上面条件时,则需双向旋转
  35. //先对右子树进行右旋转
  36. AVLNode rightSubTree = rightRotate(pointer.right);
  37. //将右旋转后新的右子树连接到该节点的右子树位置
  38. pointer.right = rightSubTree;
  39. //再对以该节点为根节点的树进行左旋转
  40. pointer = leftRotate(pointer);
  41. }else {
  42. //如果不满足条件,则直接左旋转
  43. pointer = leftRotate(pointer);
  44. }
  45. }else if (leftHeight(pointer) - rightHeight(pointer) > 1) {//判断节点的左子树高度是否比右子树高过1层以上
  46. //当右旋转时,需判断左子树的右子树高度是否高于左子树的左子树
  47. if (rightHeight(pointer.left) > leftHeight(pointer.left)) {
  48. //当满足上面条件时,则需双向旋转
  49. //先对左子树进行左旋转
  50. AVLNode leftSubTree = leftRotate(pointer.left);
  51. //将左旋转后新的左子树连接到该节点的左子树
  52. pointer.left = leftSubTree;
  53. //再对以该节点为根节点的树进行右旋转
  54. pointer = rightRotate(pointer);
  55. } else {
  56. //如果不满足条件,则直接右旋转
  57. pointer = rightRotate(pointer);
  58. }
  59. }
  60. return pointer;
  61. }

然后将原先的二叉排序树更新为添加后经过平衡的新的AVL二叉排序树

  1. //添加节点
  2. public void add(AVLNode node){
  3. //如果根节点为空则,则将传入节点设置为根节点
  4. if (root == null){
  5. root = node;
  6. }else {
  7. root = add(node, root);
  8. }
  9. }

删除操作中的平衡判断:

  1. /**
  2. * 根据value值删除节点
  3. * 删除节点可能有的3种状态:
  4. * 1.该节点是叶子节点
  5. * 2.该节点只有左子树或只有右子树
  6. * 3.该节点左子树和右子树都有
  7. * @param value 要删除节点的value值
  8. */
  9. public AVLNode delete(int value, AVLNode node){
  10. AVLNode delRes = null;//储存删除后新的树
  11. if (value < node.value){//当查找节点值小于当前节点值
  12. //向左子树递归遍历,并将删除后的新的左子树连接到左节点位置代替原先左子树
  13. node.left = delete(value, node.left);
  14. //删除后新的树
  15. delRes = node;
  16. }else if(value > node.value){//当查找节点值大于当前节点值
  17. //向右子树递归遍历,并将删除后的新的右子树连接到右节点位置代替原先右子树
  18. node.right = delete(value, node.right);
  19. //删除后新的树
  20. delRes = node;
  21. }else {//当查找节点值等于当前节点值时,即当前节点就是要删除的节点
  22. //删除节点时叶子节点的状态
  23. if (node.left == null && node.right == null) {
  24. //直接将该节点设为空
  25. //当该节点为空时,无需进入到后面的平衡判断,直接返回
  26. return null;
  27. }
  28. //删除节点左子树为空,右子树不为空的状态
  29. else if (node.left == null && node.right != null) {
  30. //保存删除节点的右子树
  31. AVLNode rightSubTree = node.right;
  32. //将删除节点的右子树设为空,使得该节点能够尽早被垃圾回收
  33. node.right = null;
  34. //删除节点的右子树,用于连接到删除节点的父节点
  35. delRes = rightSubTree;
  36. }
  37. //删除节点右子树为空,左子树不为空的状态
  38. else if (node.right == null && node.left != null) {
  39. AVLNode leftSubTree = node.left;
  40. node.left = null;
  41. delRes = leftSubTree;
  42. }
  43. //删除节点的左子树和右子树都不为空的状态
  44. //这里我们使用的是左子树的最大值节点代替的方法
  45. else {
  46. //获取左子树的最大值节点并从左子树中删除它
  47. AVLNode max = max(node.left);
  48. //将该最大值节点的左子树和右子树设置为该节点的左子树和右子树
  49. //注:删除最大值节点操作的返回值是删除后该节点的左子树
  50. max.left = delMax(node.left);
  51. max.right = node.right;
  52. //将删除节点的左子树和右子树设为空,使得该节点能够尽早被垃圾回收
  53. node.left = null;
  54. node.right = null;
  55. //执行完删除操作后,以最大值节点为根节点的新的树,用于连接的删除节点的父节点
  56. delRes = max;
  57. }
  58. }
  59.  
  60. //删除节点后要判断删除后的新的树是否平衡,如不平衡需进行旋转操作
  61. if (rightHeight(delRes) - leftHeight(delRes) > 1){
  62. if (leftHeight(delRes.right) > rightHeight(delRes.right)){
  63. AVLNode rightSubTree = rightRotate(delRes.right);
  64. node.right = rightSubTree;
  65. return leftRotate(delRes);
  66. }else {
  67. return leftRotate(delRes);
  68. }
  69. }else if (leftHeight(delRes) - rightHeight(delRes) > 1) {
  70. if (rightHeight(delRes.left) > leftHeight(delRes.left)) {
  71. AVLNode leftSubTree = leftRotate(delRes.left);
  72. delRes.left = leftSubTree;
  73. return rightRotate(delRes);
  74. } else {
  75. return rightRotate(delRes);
  76. }
  77. }
  78. //返回删除后的新树
  79. return delRes;
  80. }
  81.  
  82. /**
  83. * 查找传入节点树下value值最大的节点并删除该节点
  84. * @param node
  85. * @return
  86. */
  87. public AVLNode delMax(AVLNode node){
  88. if (node.right != null){
  89. node.right = delMax(node.right);
  90. return node;
  91. }else {
  92. AVLNode leftSubTree = node.left;
  93. node.left = null;
  94. return leftSubTree;
  95. }
  96. }
  97. /**
  98. * 查找传入节点树下value值最大的节点并放回该节点
  99. * 在二叉排序树中最大值的节点就是最右叶子节点
  100. * @param node
  101. * @return
  102. */
  103. public AVLNode max(AVLNode node){
  104. AVLNode max = node;
  105. while (max.right != null){
  106. max = max.right;
  107. }
  108. return max;
  109. }

同样也需要将原先的二叉排序树更新为删除后经过平衡的AVL二叉排序树

  1. //删除节点
  2. public void delete(int value){
  3. //判断删除节点在二叉排序树中是否存在
  4. AVLNode node = searchNode(value);
  5. if (node == null){
  6. throw new RuntimeException("二叉排序树内无对应节点");
  7. }
  8. //将删除后新的二叉排序树更换掉原先二叉排序树
  9. root = delete(value, root);
  10. }

完整代码

  1. public class AVLTreeDemo {
  2. public static void main(String[] args) {
  3. AVLTree avlTree = new AVLTree();
  4. int[] array = {4,3,6,5,7,8};//左旋转
  5. //int[] array = {10,12,8,9,7,6};//右旋转
  6. //int[] array = {10,11,7,6,8,9};//双向旋转
  7. for (int i : array){
  8. avlTree.add(new AVLNode(i));
  9. }
  10.  
  11. System.out.println("leftHeight: " + avlTree.leftHeight(avlTree.root));
  12. System.out.println("rightHeight" + avlTree.rightHeight(avlTree.root));
  13. avlTree.midOrder();
  14. }
  15. }
  16.  
  17. //AVL二叉排序树
  18. class AVLTree{
  19. AVLNode root;
  20.  
  21. public void setRoot(AVLNode root){
  22. this.root = root;
  23. }
  24.  
  25. //左旋转
  26. public AVLNode leftRotate(AVLNode node){
  27. //保存该节点的右子树
  28. AVLNode rightSubTree = node.right;
  29. //将该节点的右子树 替换成 保存的右子树rightSubTree的左子树
  30. node.right = rightSubTree.left;
  31. //再把以该节点为根节点的树 连接到 rightSubTree的左子树
  32. rightSubTree.left = node;
  33. //最后返回左旋转后新的树
  34. return rightSubTree;
  35. }
  36.  
  37. //右旋转
  38. public AVLNode rightRotate(AVLNode node){
  39. AVLNode leftSubTree = node.left;
  40. node.left = leftSubTree.right;
  41. leftSubTree.right = node;
  42. return leftSubTree;
  43. }
  44.  
  45. //获取该节点左子树的高度
  46. public int leftHeight(AVLNode node){
  47. if (node.left == null){
  48. return 0;
  49. }
  50. return height(node.left);
  51. }
  52.  
  53. //获取该节点右子树的高度
  54. public int rightHeight(AVLNode node){
  55. if (node.right == null){
  56. return 0;
  57. }
  58. return height(node.right);
  59. }
  60.  
  61. //获取以该节点为根节点的树的高度
  62. public int height(AVLNode node){
  63. /*
  64. Math.max方法可以取出俩个参数中的最大值,因为是递归叠加,所以需+1,
  65. 如当该节点是叶子节点时,如果不+1的话,会返回0,导致该节点没有被算入高度中
  66. */
  67. return Math.max(node.left==null ? 0:height(node.left), node.right==null ? 0:height(node.right)) + 1;
  68. }
  69.  
  70. //添加节点
  71. public void add(AVLNode node){
  72. //如果根节点为空则,则将传入节点设置为根节点
  73. if (root == null){
  74. root = node;
  75. }else {
  76. root = add(node, root);
  77. }
  78. }
  79.  
  80. /**
  81. * 在二叉排序树中添加节点
  82. * @param node 添加的节点
  83. * @param pointer 指针节点,用于遍历节点,初始指向根节点
  84. * @return 返回添加后平衡的新树
  85. */
  86. public AVLNode add(AVLNode node, AVLNode pointer){
  87. if (node == null){
  88. return null;
  89. }
  90.  
  91. if (pointer.value > node.value){//指针节点值大于添加节点值时
  92. //如果指针节点的左节点刚好为空,则将添加节点插入到该左节点
  93. if (pointer.left == null){
  94. pointer.left = node;
  95. }else {
  96. //如果不是则继续往左节点走
  97. pointer.left = add(node, pointer.left);
  98. }
  99. }else {//指针节点值小于添加节点值时
  100. //如果指针节点的右节点刚好为空,则将添加节点插入到该右节点
  101. if (pointer.right == null){
  102. pointer.right = node;
  103. }else {
  104. //如果不是则继续往右节点走
  105. pointer.right = add(node, pointer.right);
  106. }
  107. }
  108.  
  109. //添加完节点后,判断以该节点为根节点的树是否平衡,如不平衡则需进行旋转
  110. if (rightHeight(pointer) - leftHeight(pointer) > 1){//判断节点的右子树高度是否比左子树高过1层以上
  111. //当左旋转时,需判断右子树的左子树高度是否高于右子树的右子树
  112. if (leftHeight(pointer.right) > rightHeight(pointer.right)){
  113. //当满足上面条件时,则需双向旋转
  114. //先对右子树进行右旋转
  115. AVLNode rightSubTree = rightRotate(pointer.right);
  116. //将右旋转后新的右子树连接到该节点的右子树位置
  117. pointer.right = rightSubTree;
  118. //再对以该节点为根节点的树进行左旋转
  119. pointer = leftRotate(pointer);
  120. }else {
  121. //如果不满足条件,则直接左旋转
  122. pointer = leftRotate(pointer);
  123. }
  124. }else if (leftHeight(pointer) - rightHeight(pointer) > 1) {//判断节点的左子树高度是否比右子树高过1层以上
  125. //当右旋转时,需判断左子树的右子树高度是否高于左子树的左子树
  126. if (rightHeight(pointer.left) > leftHeight(pointer.left)) {
  127. //当满足上面条件时,则需双向旋转
  128. //先对左子树进行左旋转
  129. AVLNode leftSubTree = leftRotate(pointer.left);
  130. //将左旋转后新的左子树连接到该节点的左子树
  131. pointer.left = leftSubTree;
  132. //再对以该节点为根节点的树进行右旋转
  133. pointer = rightRotate(pointer);
  134. } else {
  135. //如果不满足条件,则直接右旋转
  136. pointer = rightRotate(pointer);
  137. }
  138. }
  139. return pointer;
  140. }
  141.  
  142. //根据value值查找节点
  143. public AVLNode searchNode(int value){
  144. if (root == null){
  145. return null;
  146. }
  147. return searchNode(value, root);
  148. }
  149.  
  150. /**
  151. * 根据value值查找节点
  152. * @param value 查找的节点
  153. * @param node 查找的树
  154. * @return
  155. */
  156. public AVLNode searchNode(int value, AVLNode node){
  157. //如果当前节点的值等于value时,则返回该节点
  158. if (node.value == value) {
  159. return node;
  160. } else if (node.value > value){//当前节点的值大于value时
  161. //如果该节点的左节点为空,则表示二叉排序树内没有该值的节点,返回空
  162. if (node.left == null)
  163. return null;
  164. //左节点不为空,继续往左子树查找
  165. return searchNode(value, node.left);
  166. }else {//当前节点的值小于value时
  167. //如果该节点的右节点为空,则表示二叉排序树内没有该值的节点,返回空
  168. if (node.right == null)
  169. return null;
  170. //右节点不为空,继续往右子树查找
  171. return searchNode(value, node.right);
  172. }
  173. }
  174.  
  175. public void delete(int value){
  176. //判断删除节点在二叉排序树中是否存在
  177. AVLNode node = searchNode(value);
  178. if (node == null){
  179. throw new RuntimeException("二叉排序树内无对应节点");
  180. }
  181. //将删除后新的二叉排序树更换掉原先二叉排序树
  182. root = delete(value, root);
  183. }
  184.  
  185. /**
  186. * 根据value值删除节点
  187. * 删除节点可能有的3种状态:
  188. * 1.该节点是叶子节点
  189. * 2.该节点只有左子树或只有右子树
  190. * 3.该节点左子树和右子树都有
  191. * @param value 要删除节点的value值
  192. */
  193. public AVLNode delete(int value, AVLNode node){
  194. AVLNode delRes = null;//储存删除后新的树
  195. if (value < node.value){//当查找节点值小于当前节点值
  196. //向左子树递归遍历,并将删除后的新的左子树连接到左节点位置代替原先左子树
  197. node.left = delete(value, node.left);
  198. //删除后新的树
  199. delRes = node;
  200. }else if(value > node.value){//当查找节点值大于当前节点值
  201. //向右子树递归遍历,并将删除后的新的右子树连接到右节点位置代替原先右子树
  202. node.right = delete(value, node.right);
  203. //删除后新的树
  204. delRes = node;
  205. }else {//当查找节点值等于当前节点值时,即当前节点就是要删除的节点
  206. //删除节点时叶子节点的状态
  207. if (node.left == null && node.right == null) {
  208. //直接将该节点设为空
  209. //当该节点为空时,无需进入到后面的平衡判断,直接返回
  210. return null;
  211. }
  212. //删除节点左子树为空,右子树不为空的状态
  213. else if (node.left == null && node.right != null) {
  214. //保存删除节点的右子树
  215. AVLNode rightSubTree = node.right;
  216. //将删除节点的右子树设为空,使得该节点能够尽早被垃圾回收
  217. node.right = null;
  218. //删除节点的右子树,用于连接到删除节点的父节点
  219. delRes = rightSubTree;
  220. }
  221. //删除节点右子树为空,左子树不为空的状态
  222. else if (node.right == null && node.left != null) {
  223. AVLNode leftSubTree = node.left;
  224. node.left = null;
  225. delRes = leftSubTree;
  226. }
  227. //删除节点的左子树和右子树都不为空的状态
  228. //这里我们使用的是左子树的最大值节点代替的方法
  229. else {
  230. //获取左子树的最大值节点并从左子树中删除它
  231. AVLNode max = max(node.left);
  232. //将该最大值节点的左子树和右子树设置为该节点的左子树和右子树
  233. //注:删除最大值节点操作的返回值是删除后该节点的左子树
  234. max.left = delMax(node.left);
  235. max.right = node.right;
  236. //将删除节点的左子树和右子树设为空,使得该节点能够尽早被垃圾回收
  237. node.left = null;
  238. node.right = null;
  239. //执行完删除操作后,以最大值节点为根节点的新的树,用于连接的删除节点的父节点
  240. delRes = max;
  241. }
  242. }
  243.  
  244. //删除节点后要判断删除后的新的树是否平衡,如不平衡需进行旋转操作
  245. if (rightHeight(delRes) - leftHeight(delRes) > 1){
  246. if (leftHeight(delRes.right) > rightHeight(delRes.right)){
  247. AVLNode rightSubTree = rightRotate(delRes.right);
  248. node.right = rightSubTree;
  249. return leftRotate(delRes);
  250. }else {
  251. return leftRotate(delRes);
  252. }
  253. }else if (leftHeight(delRes) - rightHeight(delRes) > 1) {
  254. if (rightHeight(delRes.left) > leftHeight(delRes.left)) {
  255. AVLNode leftSubTree = leftRotate(delRes.left);
  256. delRes.left = leftSubTree;
  257. return rightRotate(delRes);
  258. } else {
  259. return rightRotate(delRes);
  260. }
  261. }
  262. //返回删除后的新树
  263. return delRes;
  264. }
  265.  
  266. /**
  267. * 查找传入节点树下value值最大的节点并删除该节点
  268. * @param node
  269. * @return
  270. */
  271. public AVLNode delMax(AVLNode node){
  272. if (node.right != null){
  273. node.right = delMax(node.right);
  274. return node;
  275. }else {
  276. AVLNode leftSubTree = node.left;
  277. node.left = null;
  278. return leftSubTree;
  279. }
  280. }
  281. /**
  282. * 查找传入节点树下value值最大的节点并放回该节点
  283. * 在二叉排序树中最大值的节点就是最右叶子节点
  284. * @param node
  285. * @return
  286. */
  287. public AVLNode max(AVLNode node){
  288. AVLNode max = node;
  289. while (max.right != null){
  290. max = max.right;
  291. }
  292. return max;
  293. }
  294.  
  295. public void midOrder(){
  296. if (root != null){
  297. midOrder(root);
  298. }else {
  299. System.out.println("二叉顺序树为空,无法遍历");
  300. }
  301. }
  302.  
  303. //中序遍历
  304. public void midOrder(AVLNode node){
  305. if (node.left != null){
  306. midOrder(node.left);
  307. }
  308. System.out.println(node);
  309. if (node.right != null){
  310. midOrder(node.right);
  311. }
  312. }
  313. }
  314.  
  315. //AVL二叉排序树节点
  316. class AVLNode{
  317. int value;
  318. AVLNode left;
  319. AVLNode right;
  320.  
  321. public AVLNode(int value){
  322. this.value = value;
  323. }
  324.  
  325. @Override
  326. public String toString() {
  327. return "BSTNode{" +
  328. "value=" + value +
  329. '}';
  330. }
  331. }

数据结构与算法:AVL树的更多相关文章

  1. 数据结构与算法——AVL树类的C++实现

    关于AVL树的简单介绍能够參考:数据结构与算法--AVL树简单介绍 关于二叉搜索树(也称为二叉查找树)能够參考:数据结构与算法--二叉查找树类的C++实现 AVL-tree是一个"加上了额外 ...

  2. [数据结构与算法] : AVL树

    头文件 typedef int ElementType; #ifndef _AVLTREE_H_ #define _AVLTREE_H_ struct AvlNode; typedef struct ...

  3. 数据结构和算法(Golang实现)(28)查找算法-AVL树

    AVL树 二叉查找树的树高度影响了查找的效率,需要尽量减小树的高度,AVL树正是这样的树. 一.AVL树介绍 AVL树是一棵严格自平衡的二叉查找树,1962年,发明者Adelson-Velsky和La ...

  4. 【数据结构】平衡二叉树—AVL树

    (百度百科)在计算机科学中,AVL树是最先发明的自平衡二叉查找树.在AVL树中任何节点的两个子树的高度最大差别为一,所以它也被称为高度平衡树.查找.插入和删除在平均和最坏情况下都是O(log n).增 ...

  5. 数据结构(三)实现AVL树

    AVL树的定义 一种自平衡二叉查找树,中面向内存的数据结构. 二叉搜索树T为AVL树的满足条件为: T是空树 T若不是空树,则TL.TR都是AVL树,且|HL-HR| <= 1 (节点的左子树高 ...

  6. 数据结构与算法分析-AVL树

    1.AVL树是带有平衡条件的二叉查找树. 2.AVL树的每个节点高度最多相差1. 3.AVL树实现的难点在于插入或删除操作.由于插入和删除都有可能破坏AVL树高度最多相差1的特性,所以当特性被破坏时需 ...

  7. 数据结构——二叉查找树、AVL树

    二叉查找树:由于二叉查找树建树的过程即为插入的过程,所以其中序遍历一定为升序排列! 插入:直接插入,插入后一定为根节点 查找:直接查找 删除:叶子节点直接删除,有一个孩子的节点删除后将孩子节点接入到父 ...

  8. [算法] avl树实现

    大二的时候数据结构课死活没看懂的一个东东,看了2小时,敲了2小时,调了2小时... 平衡树某一节点的左右子树高度相差大于1的时候即需要调整,调整可分为四中情况 ll,rr,lr,rl其中lr,rl是由 ...

  9. 数据结构与算法—Trie树

    Trie,又经常叫前缀树,字典树等等.它有很多变种,如后缀树,Radix Tree/Trie,PATRICIA tree,以及bitwise版本的crit-bit tree.当然很多名字的意义其实有交 ...

  10. Android版数据结构与算法(六):树与二叉树

    版权声明:本文出自汪磊的博客,未经作者允许禁止转载. 之前的篇章主要讲解了数据结构中的线性结构,所谓线性结构就是数据与数据之间是一对一的关系,接下来我们就要进入非线性结构的世界了,主要是树与图,好了接 ...

随机推荐

  1. 使用kubernetes-event-exporter将k8s的事件导出到elasticsearch日志系统中

    使用kubernetes-event-exporter将k8s的事件导出到elasticsearch日志系统中 前提 版本 kubernetes v1.17.9 kubernetes-event-ex ...

  2. Spring security OAuth2.0认证授权学习第一天(基础概念-认证授权会话)

    这段时间没有学习,可能是因为最近工作比较忙,每天回来都晚上11点多了,但是还是要学习的,进过和我的领导确认,在当前公司的技术架构方面,将持续使用Spring security,暂不做Shiro的考虑, ...

  3. 正则表达式在Java中应用的三种典型场合:验证,查找和替换

    正则式在编程中常用,总结在此以备考: package regularexp; import java.util.regex.Matcher; import java.util.regex.Patter ...

  4. Zookeeper启动流程分析

    前言 之前的Zookeeper协议篇-Paxos算法与ZAB协议通过了解Paoxs算法开始,到Zab协议的两大特性:崩溃恢复和消息广播,学习了Zookeeper是如何通过Zab协议实现高可用,本篇主要 ...

  5. CentOS 7 安装部署 cassandra作为kairosdb的数据存储

    环境 Centos 7.4 java 1.8.0 安装步骤 java yum -y install java-1.8.0-openjdk* cassandra wget https://mirrors ...

  6. 微信小程序常用样式

    1.设置全局字体样式app.wxss: text{ font-family:MicroSoft yahei; } 2.设置弹性盒子模型: .container{ /*弹性模型*/ display:fl ...

  7. oracle之insert语句总结

    insert语句总结 16.1 第一类,insert语句:单行插入 1)SQL> create table a (id int,name char(10) default 'aaa');   / ...

  8. docker部署LAMP架构并部署上线wordpress博客系统

    第一步:直接在镜像仓库拉取LAMP镜像 [root@ken-node3 ken]# docker pull tutum/lamp 第二步:查看已经获取到的镜像 [root@ken-node3 ken] ...

  9. lombok使用(给自己看的,只为不要忘记自己用过的技术)

    如何使用? 一.1)eclipse使用方法 1. 从项目首页下载lombok.jar 2. 双击lombok.jar, 将其安装到eclipse中(该项目需要jdk1.6+的环境) 2)idea使用方 ...

  10. 如何使用NuGet package .nupkg文件?

    如果你本来就有.nupkg文件并且你只需要.dll文件的话,你可以通过打开.zip下的lib文件夹来获取. 例如: