c++-AVL树

文章目录

  • 前言
  • 一、AVL树
    • 1、AVL树概念
    • 2、AVL树模拟实现
    • 3、AVL树的旋转操作
      • 3.1 左单旋
      • 3.2 左单旋代码实现
      • 3.3 右单旋
      • 3.4 右单旋代码实现。
      • 3.5 什么时候调用左单旋和右单旋
      • 3.6 左右双旋
      • 3.7 左右双旋代码实现
      • 3.8 右左双旋
      • 3.9 右左双旋代码实现
      • 3.10 什么时候调用左右双旋和右左双旋
    • 4、测试
    • 5、总结
    • 6、AVL树的性能
    • 7、AVL树的结点删除
    • 8、代码


前言


一、AVL树

1、AVL树概念

前面对map/multimap/set/multiset进行了简单的介绍,在其文档介绍中发现,这几个容器有个共同点是:其底层都是按照二叉搜索树来实现的,但是二叉搜索树有其自身的缺陷,假如往树中插入的元素有序或者接近有序,二叉搜索树就会退化成单支树,时间复杂度会退化成O(N),因此map、set等关联式容器的底层结构是对二叉树进行了平衡处理,即采用平衡树来实现。

二叉搜索树虽可以缩短查找的效率,但如果数据有序或接近有序二叉搜索树将退化为单支树,查找元素相当于在顺序表中搜索元素,效率低下。因此,两位俄罗斯的数学家G.M.Adelson-Velskii和E.M.Landis在1962年发明了一种解决上述问题的方法:当向二叉搜索树中插入新结点后,如果能保证每个结点的左右子树高度之差的绝对值不超过1(需要对树中的结点进行调整),即可降低树的高度,从而减少平均搜索长度。

一棵AVL树或者是空树,或者是具有以下性质的二叉搜索树:

  • 它的左右子树都是AVL树
  • 左右子树高度之差(简称平衡因子)的绝对值不超过1(-1/0/1)
  • 如果一棵二叉搜索树是高度平衡的,它就是AVL树。如果它有n个结点,其高度可保持在 O ( l o g 2 n ) O(log_2 n) O(log2n),搜索时间复杂度O( l o g 2 n log_2 n log2n)。

在这里插入图片描述

2、AVL树模拟实现

下面我们求AVL树的平衡因子采用的是右子树高度减左子树高度。
我们将AVL树的结点定义为下面这样

	template<class K, class V>class AVLTreeNode{AVLTreeNode<K, V>* _left;   //当前结点左孩子AVLTreeNode<K, V>* _right;  //当前结点右孩子AVLTreeNode<K, V>* _parent;  //当前结点父结点pair<K, V> _kv;int _bf;   //平衡因子AVLTreeNode(const pair<K, V>& kv):_left(nullptr), _right(nullptr), _parent(nullptr), _kv(kv), _bf(0){}};

AVL树就是在二叉搜索树的基础上引入了平衡因子,因此AVL树也可以看成是二叉搜索树。下面我们先来实现二叉搜索树的插入结点的方法。

template<class K, class V>class AVLTree{typedef AVLTreeNode<K, V> Node;public:bool Insert(const pair<K, V>& kv){if (_root == nullptr){_root = new Node(kv);return true;}Node* parent = nullptr;Node* cur = _root;while (cur){if (cur->_kv.first < kv.first){parent = cur;cur = cur->_right;}else if (cur->_kv.first > kv.first){parent = cur;cur = cur->_left;}else{return false;}}//找到结点要插入的位置,将结点链接到二叉搜索树中cur = new Node(kv);if (parent->_kv.first < cur->_kv.first){parent->_right = cur;}else{parent->_left = cur;}//将父结点也改变cur->_parent = parent;return true;}private:Node* _root = nullptr;};
}

上面的代码中我们就初步完成了二叉搜索树的结点插入,因为AVL树的插入需要保持平衡因子,所以我们在插入新结点时还需要来维护平衡因子。
当插入新结点cur后,我们就需要更新parent的平衡因子。在插入cur之前,parent的平衡因子分为三种情况:-1,0,1。
(1). 如果cur插入到parent的左侧,只需给parent的平衡因子-1即可。即parent->_bf- -;
(2). 如果cur插入到parent的右侧,只需给parent的平衡因子+1即可。即parent->_bf++;

当更新完parent的平衡因子后,可能会影响parent的祖先的平衡因子(全部祖先或部分祖先),parent的平衡因子可能会出现下面的三种情况:
(1). 如果parent的平衡因子为0,说明插入之前parent的平衡因子为正1或负1,插入后被调整为0,此时还满足AVL树的性质,所以当前cur结点插入成功。此时以parent为根的树的高度不变,不用继续往上更新。
(2). 如果parent的平衡因子为正1或负1,说明插入前parent的平衡因子一定为0,插入后被更新成正1或负1,此时以parent为根的树的高度增加,需要继续向上更新父结点的平衡因子。
(3). 如果parent的平衡因子为正2或负2,则parent的平衡因子违反平衡树的性质,需要对其进行旋转处理。
在这里插入图片描述
下面我们来使用代码实现上面的分析过程。

bool Insert(const pair<K, V>& kv){if (_root == nullptr){_root = new Node(kv);return true;}Node* parent = nullptr;Node* cur = _root;while (cur){if (cur->_kv.first < kv.first){parent = cur;cur = cur->_right;}else if (cur->_kv.first > kv.first){parent = cur;cur = cur->_left;}else{return false;}}//找到结点要插入的位置,将结点链接到二叉搜索树中cur = new Node(kv);if (parent->_kv.first < cur->_kv.first){parent->_right = cur;}else{parent->_left = cur;}//将父结点也改变cur->_parent = parent;//更新平衡因子while (parent){//先判断cur插入到parent的左边或右边,然后更新parent的平衡因子if (cur == parent->_right){parent->_bf++;}else{parent->_bf--;}//判断更新后parent的平衡因子为多少,是否还需要继续向上更新祖先的平衡因子if (parent->_bf == 1 || parent->_bf == -1){//说明以parent为根的树的高度改变,需要更新parent祖先的平衡因子parent = parent->_parent;cur = cur->_parent;}else if (parent->_bf == 0){//说明以parent为根的树的高度不变,不需要更新parent祖先的平衡因子,插入结点成功break;}else if (parent->_bf == 2 || parent->_bf == -2){//说明parent的平衡因子违反AVL树的性质,需要对其进行旋转处理,使其符合AVL树性质break;}else{//如果不是上面的任何一种情况,也报错误。assert(false);}}return true;}

3、AVL树的旋转操作

3.1 左单旋

当新结点插入较高右子树的右侧时,即在c子树中插入新结点时就会引发30结点的平衡因子不符合AVL树性质。
在这里插入图片描述

如果出现上面的情况时,我们就可以使用左单旋操作来将以30结点为根结点的这棵二叉树变为AVL树。下面的图中画了左单旋的具体操作。
在这里插入图片描述

3.2 左单旋代码实现

下面我们来实现左单旋操作的代码。
在这里插入图片描述

我们将左单旋的代码这样实现,逻辑上是对的,但是因为我们定义AVL树的结点为三叉链,而下面的代码中没有维护每个结点的 _ parent指针,这显然是不行的。
在这里插入图片描述
下面我们将每个结点的 _ parent指针也进行更新,但是代码中还是存在问题。即当h == 0时,subRL为nullptr,然后代码中的subRL -> _ parent就会出现错误。并且在实际中,parent不一定为根结点,所以我们还需要提前设置一个pparent指针记录parent结点的父结点,然后判断parent结点是否为根结点,如果不是根结点就需要将subR为pparent的孩子,pparent为subR的父结点。并且我们还需要记得更新每个结点的平衡因子,我们看到左单旋的过程中,parent指向的结点和subR指向的结点的平衡因子都变为0了,而其它结点的平衡因子没有变化。
在这里插入图片描述
在这里插入图片描述
下面就是左单旋的完整代码,可以体会到左单旋的代码实现还是要注意很多细节的。

//左单旋void RotateL(Node* parent){Node* subR = parent->_right;Node* subRL = subR->_left;parent->_right = subRL;//如果subRL不为nullptr,才将subRL指向的结点的_parent指针更新if (subRL != nullptr){subRL->_parent = parent;}//提前记录parent指向结点的父结点的指针。Node* pparent = parent->_parent;subR->_left = parent;parent->_parent = subR;//判断parent是否为根结点,因为只有根结点没有父亲if (pparent == nullptr){//如果parent为根结点,那么左单旋后subR为新的根结点_root = subR;_root->_parent = nullptr;}else{//parent不为根结点,那么就将subR更新为pparent的孩子//判断将subR为pparent的左孩子还是右孩子if (pparent->_left = parent){pparent->_left = subR;}else{pparent->_right = subR;}subR->_parent = pparent;}//更新结点平衡因子parent->_bf = 0;subR->_bf = 0;}

3.3 右单旋

下面我们再来分析右单旋的情况。当新节点插入较高左子树的左侧时,即在a子树中插入新结点时就会引发60结点的平衡因子不符合AVL树性质。
在这里插入图片描述
如果出现上面的情况时,我们就可以使用右单旋操作来将以60结点为根结点的这棵二叉树变为AVL树。下面的图中画了右单旋的具体操作。
在这里插入图片描述

3.4 右单旋代码实现。

前面我们已经实现了左单旋的代码,那么右单旋的代码实现和左单旋类似,我们一样需要注意上面的一些细节。
在这里插入图片描述

//右单旋void RotateR(Node* parent){Node* subL = parent->_left;Node* subLR = subL->_right;parent->_left = subLR;//判断subLR是否为空结点,如果为空结点就不需要更新_parent指针if (subLR != nullptr){subLR->_parent = parent;}//提前记录parent指向的结点的父结点指针Node* pparent = parent->_parent;subL->_right = parent;parent->_parent = subL;//我们换一个方法判断parent是否为根结点//如果parent为根结点,就将subL设置为新的根结点if (parent == _root){_root = subL;_root->_parent = nullptr;}else{//判断subL应该作为pparent指向结点的左孩子还是右孩子if (pparent->_left == parent){pparent->_left = subL;}else{pparent->_right = subL;}subL->_parent = pparent;  //更新subL_parent指针}//更新subL和parent结点的平衡因子subL->_bf = 0;parent->_bf = 0;}

3.5 什么时候调用左单旋和右单旋

我们已经实现了左单旋和右单旋的代码,那么我们应该在什么使用调用左单旋函数,什么时候调用右单旋函数呢?
当新节点插入较高右子树的右侧时,如下图所示的情况,我们此时调用左单旋。
在这里插入图片描述
当新节点插入较高左子树的左侧时,如下图所示的情况,我们此时调用右单旋。
在这里插入图片描述

代码中实现判断
在这里插入图片描述

3.6 左右双旋

下面我们再来分析左右双旋的情况。前面我们分析了当新节点插入较高左子树的左侧时,即在a子树中插入新结点时就会引发下图中90结点的平衡因子不符合AVL树性质,此时需要将30结点进行右单旋。而当我们将新节点插入较高左子树的右侧时,此时将30结点进行右单旋后并不能将这棵树变为AVL树。
在这里插入图片描述
如果出现上面的情况时,我们就需要使用左右双旋操作来将以90结点为根结点的这棵二叉树变为AVL树,即我们先将30结点左单旋,然后再将90结点右单旋。下面的图中画了左右双旋的具体操作。
在这里插入图片描述
从上面的过程中我们可以看到左右双旋其实就是让60结点去上面做根结点,然后60结点的左右孩子分别给30结点和90结点。

3.7 左右双旋代码实现

因为左右双旋分开的话其实就是第一步左单旋,第二步右单旋,所以我们可以复用前面写的左单旋和右单旋函数来实现左右双旋。
在这里插入图片描述

那么只需要这样就实现了左右双旋吗?
显然下面的代码不完整,因为我们通过上面的分析看到了左右双旋中最麻烦的其实是对结点的平衡因子进行更新,因为我们发现了这三个结点的平衡因子共出现了三种情况。
在这里插入图片描述
当在c子树中插入新结点时,此时subLR的平衡因子为1,左右双旋后三个结点的平衡因子分别为:

parent->_bf = 0
subL->_bf = -1
subLR->_bf = 0

在这里插入图片描述

当在b子树中插入新结点时,此时subLR的平衡因子为-1,左右双旋后三个结点的平衡因子分别为:

parent->_bf = 1
subL->_bf = 0
subLR->_bf = 0

在这里插入图片描述
然后还有最后一种情况,就是h为0时,此时60结点就是新插入的结点,subLR的平衡因子为0。左右双旋后三个结点的平衡因子分别为:

parent->_bf = 0
subL->_bf = 0
subLR->_bf = 0

在这里插入图片描述
下面我们来写代码实现。

//左右双旋void RotateLR(Node* parent){Node* subL = parent->_left;Node* subLR = subL->_right;//用来判断平衡因子为哪种情况int bf = subLR->_bf;RotateL(parent->_left);RotateL(parent);if (bf == 1){parent->_bf = 0;subL->_bf = -1;subLR->_bf = 0}else if (bf == -1){parent->_bf = 1;subL->_bf = 0;subLR->_bf = 0;}//我们不建议直接将bf==0的情况写在else里面,因为我们可以将else后使用一个断言,用来发现其它不可预知的情况//即判断走到了else时,说明程序出现了大问题,断言可以更好的帮我们检查出问题。else if (bf == 0){parent->_bf = 0;subL->_bf = 0;subLR->_bf = 0;}else{assert(false);}}

3.8 右左双旋

我们前面分析了当新结点插入较高右子树的右侧时,即在d子树中插入新结点时就会引发30结点的平衡因子不符合AVL树性质,此时30结点可以通过左单旋来进行调整。而当我们将新节点插入较高右子树的左侧时,即在b或c子树中插入新结点,此时左单旋并不能将这棵子树调整为AVL树,此时就需要进行右左双旋操作来调整AVL树。
在这里插入图片描述
如果出现上面的情况时,我们就需要使用右左双旋操作来将以30结点为根结点的这棵二叉树变为AVL树,即我们先将90结点右单旋,然后再将30结点左单旋。下面的图中画了右左双旋的具体操作。(纠正:下图中的60结点右单旋都改为90结点右单旋)。
在这里插入图片描述

3.9 右左双旋代码实现

前面我们已经实现了左右双旋的代码,右左双旋的实现也是需要注意每个结点的平衡因子的更新。
在这里插入图片描述
当我们在b子树中插入新结点时,此时subRL的平衡因子为-1,左右双旋后三个结点的平衡因子分别为:

parent->_bf = 0
subR->_bf = 1
subRL->_bf = 0

在这里插入图片描述
当我们在c子树中插入新结点时,此时subRL的平衡因子为1,左右双旋后三个结点的平衡因子分别为:

parent->_bf = -1
subR->_bf = 0
subRL->_bf = 0

在这里插入图片描述

还有最后一种情况,就是h为0时,此时60结点就是新插入的结点,subRL的平衡因子为0。左右双旋后三个结点的平衡因子分别为:

parent->_bf = 0
subR->_bf = 0
subRL->_bf = 0

在这里插入图片描述
下面我们使用代码来实现右左双旋。

//右左双旋void RotateRL(Node* parent){Node* subR = parent->_right;Node* subRL = subR->_left;int bf = subRL->_bf;RotateR(parent->_right);RotateL(parent);if (bf == 1){parent->_bf = -1;subR->_bf = 0;subRL->_bf = 0;}else if (bf == -1){parent->_bf = 0;subR->_bf = 1;subRL->_bf = 0;}else if (bf == 0){parent->_bf = 0;subR->_bf = 0;subRL->_bf = 0;}else{assert(false);}}

3.10 什么时候调用左右双旋和右左双旋

我们实现了左右双旋和右左双旋的代码后,下面我们就要分析在什么情况下会调用这两个函数了。
当新节点插入较高左子树的右侧时,如下图所示的情况,我们使用左右双旋,即先让30结点进行左单旋,然后让90结点进行右单旋。
在这里插入图片描述

当新节点插入较高右子树的左侧时,如下图所示的情况,我们使用右左双旋,即先让90结点进行右单旋,然后让30结点进行左单旋。
在这里插入图片描述

代码中实现判断
在这里插入图片描述

4、测试

上面我们就实现了一个AVL树,下面我们来进行测试。
我们知道AVL树是一棵平衡二叉搜索树,下面我们写一个中序遍历,然后我们创建一个AVL树,并且插入数据,然后中序遍历这棵AVL树,看输出的结果是否为升序。
在这里插入图片描述
我们看到输出的结果和我们预期的一样为升序,但是这样就可以判断我们实现的AVL树没有问题了吗?
这样的测试肯定是不够的,而且AVL树是一棵平衡二叉搜索树,这个结果只能说明我们实现了一棵二叉搜索树,这个二叉搜索树是否为平衡的,我们不能得出结论。
在这里插入图片描述
所以我们需要写一个方法来判断这棵二叉搜索树是否为一棵平衡树。我们知道平衡二叉搜索树的每个结点的平衡因子的绝对值不大于1,那么如果我们直接检查每个结点的平衡因子是否有绝对值大于1的,这个方法可以不可以呢?这样做也是不严谨的,因为平衡因子是我们自己维护并且更新的,如果我们的逻辑错误,那么可能造成平衡因子正确,但是二叉搜索树不平衡的情况发生。所以我们还是需要通过一一检查每个结点的左右子树的高度差是否大于1这样的方法来判断。
我们先写一个函数来求一棵树的高度。
在这里插入图片描述
然后我们再写一个函数递归判断每一个结点是否都满足左右子树高度差小于2,如果每一个结点都满足那么这棵树就是平衡二叉搜索树了。
在这里插入图片描述
但是有时候可能我们的逻辑错误,使AVL树没问题,但是平衡因子更新时出现了问题,虽然此时还是一棵AVL树,但是因为平衡因子不对,就会导致以后再使用时出现问题。所以我们可以在判断每个结点是否都满足平衡树的同时也判断每个结点的平衡因子是否都正确。这样我们的判断才严谨。并且这样处理也方便以后出问题了调试时我们可以更快的定位到错误。
在这里插入图片描述
下面我们将右左双旋中的平衡因子更新代码注释掉,我们可以看到结果中显示6结点的平衡因子异常,并且此时二叉搜索树也不是平衡二叉搜索树了。
在这里插入图片描述
在这里插入图片描述
6结点出现错误并不一定就是插入6结点时引起的错误,此时我们需要一个结点一个结点的去排查,但是这样排查也会很慢,此时我们可以每插入一个结点就使用IsBalance函数判断当前的二叉搜索树是否为平衡二叉搜索树。这样我们就很快的看到了是14结点插入时出现了问题。
在这里插入图片描述
此时我们就可以自己手动写一个判断条件,然后在判断条件里面打一个断点,这样来快速的还原错误bug现场。这样我们的调试效率就可以加快了。
在这里插入图片描述
下面我们将右左双旋更新平衡因子的代码恢复,然后我们写一个生成随机数的代码来再次测试我们实现的AVL是否还有问题。这样多执行几次,就基本将所有的场景都测试到了。
在这里插入图片描述

5、总结

AVL树插入结点引起的这棵树不平衡情况可以通过下面的图来记忆。
在这里插入图片描述

6、AVL树的性能

AVL树是一棵绝对平衡的二叉搜索树,其要求每个节点的左右子树高度差的绝对值都不超过1,这样可以保证查询时高效的时间复杂度,即 l o g 2 ( N ) log_2 (N) log2(N)。但是如果要对AVL树做一些结构修改的操作,性能非常低下,比如:插入时要维护其绝对平衡,旋转的次数比较多,更差的是在删除时,有可能一直要让旋转持续到根的位置。因此:如果需要一种查询高效且有序的数据结构,而且数据的个数为静态的(即不会改变),可以考虑AVL树,但一个结构经常修改,就不太适合。

7、AVL树的结点删除

AVL树的结点删除后也可能会使这棵树变得不平衡,此时也需要通过旋转操作来重新使这棵树满足平衡二叉搜索树的性质。
与AVL树插入结点不同的是,AVL树中删除结点后,如果父结点平衡因子更新为1或-1时,此时不需要继续向上更新祖先结点的平衡因子,直接就是成功删除结点了。因为说明父结点之前的平衡因子为0,此时删除了结点后子树高度没有变化,就不会使AVL树变得不平衡了。
在这里插入图片描述

如果删除结点后,该结点的父结点平衡因子更新为0,那么就需要继续向上更新祖先结点的平衡因子,因为说明原来父结点的平衡因子为1或-1,删除结点后父结点的平衡因子才变为0。子树高度发生变化了。并且在向上继续更新祖先结点的平衡因子的时候,如果祖先结点的平衡因子不满足AVL树的要求,那么就需要通过旋转操作来将这棵树修正为平衡二叉搜索树。
在这里插入图片描述

8、代码

#pragma once#include<iostream>
#include<map>
#include<utility>
#include<assert.h>
#include<utility>
#include<time.h>using std::pair;
using std::cout;
using std::endl;namespace dong
{template<class K, class V>struct AVLTreeNode{AVLTreeNode<K, V>* _left;   //当前结点左孩子AVLTreeNode<K, V>* _right;  //当前结点右孩子AVLTreeNode<K, V>* _parent;  //当前结点父结点pair<K, V> _kv;int _bf;   //平衡因子AVLTreeNode(const pair<K, V>& kv):_left(nullptr), _right(nullptr), _parent(nullptr), _kv(kv), _bf(0){}};template<class K, class V>class AVLTree{typedef AVLTreeNode<K, V> Node;public:bool Insert(const pair<K, V>& kv){if (_root == nullptr){_root = new Node(kv);return true;}Node* parent = nullptr;Node* cur = _root;while (cur){if (cur->_kv.first < kv.first){parent = cur;cur = cur->_right;}else if (cur->_kv.first > kv.first){parent = cur;cur = cur->_left;}else{return false;}}//找到结点要插入的位置,将结点链接到二叉搜索树中cur = new Node(kv);if (parent->_kv.first > kv.first){parent->_left = cur;}else{parent->_right = cur;}//将父结点也改变cur->_parent = parent;//更新平衡因子while (parent){//先判断cur插入到parent的左边或右边,然后更新parent的平衡因子if (cur == parent->_right){parent->_bf++;}else{parent->_bf--;}//判断更新后parent的平衡因子为多少,是否还需要继续向上更新祖先的平衡因子if (parent->_bf == 1 || parent->_bf == -1){//说明以parent为根的树的高度改变,需要更新parent祖先的平衡因子parent = parent->_parent;cur = cur->_parent;}else if (parent->_bf == 0){//说明以parent为根的树的高度不变,不需要更新parent祖先的平衡因子,插入结点成功break;}else if (parent->_bf == 2 || parent->_bf == -2){//说明parent的平衡因子违反AVL树的性质,需要对其进行旋转处理,使其符合AVL树性质//需要旋转处理: 1、让这棵子树平衡  2、降低这棵子树的高度//调用左单旋if (parent->_bf == 2 && cur->_bf == 1){RotateL(parent);}//调用右单旋else if (parent->_bf == -2 && cur->_bf == -1){RotateR(parent);}//左右双旋else if(parent->_bf == -2 && cur->_bf == 1){RotateLR(parent);}//右左双旋else if (parent->_bf == 2 && cur->_bf == -1){RotateRL(parent);}else{assert(false);}break;}else{//如果不是上面的任何一种情况,也报错误。assert(false);}}return true;}void InOrder(){_InOrder(_root);cout << endl;}bool IsBalance(){return _IsBalance(_root);}protected://左单旋void RotateL(Node* parent){Node* subR = parent->_right;Node* subRL = subR->_left;parent->_right = subRL;//如果subRL不为nullptr,才将subRL指向的结点的_parent指针更新if (subRL != nullptr){subRL->_parent = parent;}//提前记录parent指向结点的父结点的指针。Node* pparent = parent->_parent;subR->_left = parent;parent->_parent = subR;//判断parent是否为根结点,因为只有根结点没有父亲if (pparent == nullptr){//如果parent为根结点,那么左单旋后subR为新的根结点_root = subR;_root->_parent = nullptr;}else{//parent不为根结点,那么就将subR更新为pparent的孩子//判断将subR为pparent的左孩子还是右孩子if (pparent->_left == parent){pparent->_left = subR;}else{pparent->_right = subR;}subR->_parent = pparent;}//更新结点平衡因子parent->_bf = 0;subR->_bf = 0;}//右单旋void RotateR(Node* parent){Node* subL = parent->_left;Node* subLR = subL->_right;parent->_left = subLR;//判断subLR是否为空结点,如果为空结点就不需要更新_parent指针if (subLR != nullptr){subLR->_parent = parent;}//提前记录parent指向的结点的父结点指针Node* pparent = parent->_parent;subL->_right = parent;parent->_parent = subL;//我们换一个方法判断parent是否为根结点//如果parent为根结点,就将subL设置为新的根结点if (parent == _root){_root = subL;_root->_parent = nullptr;}else{//判断subL应该作为pparent指向结点的左孩子还是右孩子if (pparent->_left == parent){pparent->_left = subL;}else{pparent->_right = subL;}subL->_parent = pparent;  //更新subL_parent指针}//更新subL和parent结点的平衡因子subL->_bf = 0;parent->_bf = 0;}//左右双旋void RotateLR(Node* parent){Node* subL = parent->_left;Node* subLR = subL->_right;//用来判断平衡因子为哪种情况int bf = subLR->_bf;RotateL(parent->_left);RotateR(parent);//更新平衡因子if (bf == 1){parent->_bf = 0;subL->_bf = -1;subLR->_bf = 0;}else if (bf == -1){parent->_bf = 1;subL->_bf = 0;subLR->_bf = 0;}//我们不建议直接将bf==0的情况写在else里面,因为我们可以将else后使用一个断言,用来发现其它不可预知的情况//即判断走到了else时,说明程序出现了大问题,断言可以更好的帮我们检查出问题。else if (bf == 0){parent->_bf = 0;subL->_bf = 0;subLR->_bf = 0;}else{assert(false);}}//右左双旋void RotateRL(Node* parent){Node* subR = parent->_right;Node* subRL = subR->_left;int bf = subRL->_bf;RotateR(parent->_right);RotateL(parent);//更新平衡因子if (bf == 1){parent->_bf = -1;subR->_bf = 0;subRL->_bf = 0;}else if (bf == -1){parent->_bf = 0;subR->_bf = 1;subRL->_bf = 0;}else if (bf == 0){parent->_bf = 0;subR->_bf = 0;subRL->_bf = 0;}else{assert(false);}}void _InOrder(Node* root){if (root == nullptr){return;}_InOrder(root->_left);cout << root->_kv.first << " ";_InOrder(root->_right);}//得到一棵树的高度int _Height(Node* root){if (root == nullptr){return 0;}int leftH = _Height(root->_left);int rightH = _Height(root->_right);return leftH > rightH ? leftH + 1 : rightH + 1;}bool _IsBalance(Node* root){//如果是空树则也为平衡树if (root == nullptr){return true;}//求结点的左子树高度int leftH = _Height(root->_left);//求结点的右子树高度int rightH = _Height(root->_right);//判断平衡因子是否正确if (rightH - leftH != root->_bf){cout << root->_kv.first << "结点平衡因子异常" << endl;return false;}//每一个结点都检查是否满足平衡树return abs(leftH - rightH) < 2&& _IsBalance(root->_left)&& _IsBalance(root->_right);}private:Node* _root = nullptr;};
}void Test01()
{//int a[] = { 16,3,7,11,9,26,18,14,15 };int a[] = { 4, 2, 6, 1, 3, 5, 15, 7, 16, 14 };dong::AVLTree<int, int> t1;for (auto e : a){if (e == 14){//只是用来打断点用int x = 0;}t1.Insert(std::make_pair(e,e));cout << e << "插入:" << t1.IsBalance() << endl;}t1.InOrder();cout << t1.IsBalance() << endl;
}void Test02()
{srand(time(0));const size_t N = 100000;dong::AVLTree<int, int> t;//生成随机数插入到AVL树中for (size_t i = 0; i < N; ++i){size_t x = rand();t.Insert(std::make_pair(x, x));}//看最后是否还满足AVL树cout << t.IsBalance() << endl;
}int main()
{//Test01();Test02();return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/157930.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ROS Hello World

万物始于Hello World&#xff0c;为了体验ROS&#xff0c;使用Hello World介绍ROS的简单使用。 一、Hello World工程简介 首先需要创建工程&#xff0c;流程为&#xff1a; 创建工作空间目录&#xff08;即工程根目录&#xff0c;注意此时还不是ROS工作空间&#xff0c;只是…

Spring Boot Web MVC

文章目录 一、Spring Boot Web MVC 概念二、状态码三、其他注解四、响应操作 一、Spring Boot Web MVC 概念 Spring Web MVC 是⼀个 Web 框架&#xff0c;一开始就包含在Spring 框架里。 1. MVC 定义 软件⼯程中的⼀种软件架构设计模式&#xff0c;它把软件系统分为模型、视…

Nginx域名重定向(如何访问的域名和实际的数据请求路径不同,可解决前端跨域)

感情需要被抑制&#xff0c;不能泛滥… 当需要将一个域名重定向到另一个域名并且用户仍然看到原始域名时&#xff0c;Nginx是一个强大的工具。这种场景通常涉及到反向代理或重写URL的技巧。在本篇博客中&#xff0c;我们将详细介绍如何使用Nginx来实现这个目标&#xff0c;以及…

GitLab(2)——Docker方式安装Gitlab

目录 一、前言 二、安装Gitlab 1. 搜索gitlab-ce镜像 2. 下载镜像 3. 查看镜像 4. 提前创建挂载数据卷 5. 运行镜像 三、配置Gitlab文件 1. 配置容器中的/etc/gitlab/gitlab.rb文件 2. 重启容器 3. 登录Gitalb ① 查看初始root用户的密码 ② 访问gitlab地址&#…

HTML区块、布局

HTML区块&#xff1a; HTML可以通过<div> 和 <span>将元素组合起来。大多数HTML元素被定义为块级元素或内联元素。块级元素在浏览器显示时&#xff0c;通常会以新行来开始、结束&#xff1b;内联元素在显示时通常不会以新行开始。 HTML<div>元素是块级元素…

记一次 logback 没有生成独立日志文件问题

背景 在新项目发布后发现日志文件并没有按照期望的方式独立开来&#xff0c;而是都写在了 application.log 文件中。 问题展示 日志文件&#xff1a; 项目引入展示&#xff1a; <include resource"paas/sendinfo/switch/client/sendinfo-paas-switch-client-log.…

正点原子嵌入式linux驱动开发——Linux 多点电容触摸屏

随着智能手机的发展&#xff0c;电容触摸屏也得到了飞速的发展。相比电阻触摸屏&#xff0c;电容触摸屏有很多的优势&#xff0c;比如支持多点触控、不需要按压&#xff0c;只需要轻轻触摸就有反应。ALIENTEK的三款RGB LCD屏幕都支持多点电容触摸&#xff0c;本章就以ATK7016这…

服务器数据恢复-VSAN环境下ESXI虚拟机无法访问的数据恢复方案

一、用户信息&#xff1a; 广东某单位 二、数据恢复环境&#xff1a; 主机操作系统&#xff1a;ESXI 分区类型&#xff1a;VSAN 存储介质清单 &#xff1a;一共8台服务器节点&#xff0c;每节点2个磁盘组&#xff0c;其中1个磁盘组配置1块SSD固态硬盘&#xff0c;4块1.2T机…

如何在MacOS使用homebrew安装Nginx

文章目录 Homebrew安装nginxbrew启动Nginxbrew关闭Nginx 参考文章地址 Homebrew安装nginx 在确保MacOS 安装Homebrew成功以后&#xff0c;执行如下命令 brew install nginx注意&#xff1a;Nginx安装成功后会提示目录位置&#xff1b;每个人的系统可能因为Homebrew的安装位置…

自定义的卷积神经网络模型CNN,对图片进行分类并使用图片进行测试模型-适合入门,从模型到训练再到测试,开源项目

自定义的卷积神经网络模型CNN&#xff0c;对图片进行分类并使用图片进行测试模型-适合入门&#xff0c;从模型到训练再到测试&#xff1a;开源项目 开源项目完整代码及基础教程&#xff1a; https://mbd.pub/o/bread/ZZWclp5x CNN模型&#xff1a; 1.导入必要的库和模块&…

前端接口请求支持内容缓存和过期时间

前端接口请求支持内容缓存和过期时间 支持用户自定义缓存时间&#xff0c;在规则时间内读取缓存内容&#xff0c;超出时间后重新请求接口 首先封装一下 axios&#xff0c;这一步可做可不做。但是在实际开发场景中都会对 axios 做二次封装&#xff0c;我们在二次封装的 axios …

k8s之亲和性、污点

目录 亲和性 键值运算关系 硬策略 软策略 Pod亲和性与反亲和性 污点(Taint) 和 容忍(Tolerations) 污点(Taint) 容忍(Tolerations) 维护操作 故障排除步骤 亲和性 官方介绍&#xff1a;https://kubernetes.io/zh/docs/concepts/scheduling-eviction/assign-pod-nod…