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.rhkb.cn/news/178646.html

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

相关文章

Kafka - 监控工具 Kafka Eagle:实时洞察Kafka集群的利器

文章目录 引言Kafka Eagle简介Kafka Eagle的特点Kafka Eagle的优势使用Kafka Eagle的步骤结论 引言 在现代大数据架构中&#xff0c;Apache Kafka已成为一个不可或缺的组件&#xff0c;用于可靠地处理和传输大规模的数据流。然而&#xff0c;随着Kafka集群规模的不断增长&…

编写shell脚本,利用mysqldump实现MySQL数据库分库分表备份

查看数据和数据表 mysql -uroot -p123456 -e show databases mysql -uroot -p123456 -e show tables from cb_d 删除头部Database和数据库自带的表 mysql -uroot -p123456 -e show databases -N | egrep -v "information_schema|mysql|performance_schema|sys"编写…

Linux之sched_setscheduler调度策略总结(六十)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

设计模式第一课-单例模式(懒汉模式和饿汉模式)

单例模式 个人理解&#xff1a;单例模式实际就是通过类加载的方式获取到一个对象&#xff0c;并且保证这个对象在使用中只有一个&#xff0c;不允许再次被创建 一、懒汉模式 1、懒汉模式的基础写法 代码解释&#xff1a; &#xff08;1&#xff09;、编写LazySingleton类的…

Shiny Server和ShinyProxy是什么,有什么区别?

调研以及参与过多个生物公司的生信工具研发&#xff0c;不管是ShinyServer还是ShinyProxy都有一定研究&#xff0c;尤其是ShinyServer。如果仅是本地化测试想快速的搭建Shiny应用&#xff0c;我推荐用Shiny Server&#xff0c;如果多并发用户且更好的线上管理Shiny应用&#xf…

React Native 样式及其布局

React Native 样式及其布局 参考 https://reactnative.cn/docs/flexbox 一、样式 在 React Native 中&#xff0c;你并不需要学习什么特殊的语法来定义样式。我们仍然是使用 JavaScript 来写样式。所有的核心组件都接受名为style的属性。这些样式名基本上是遵循了 web 上的 …

安全与HTTP协议:为何明文传输数据成为争议焦点?

&#x1f3ac; 江城开朗的豌豆&#xff1a;个人主页 &#x1f525; 个人专栏 :《 VUE 》 《 javaScript 》 &#x1f4dd; 个人网站 :《 江城开朗的豌豆&#x1fadb; 》 ⛺️ 生活的理想&#xff0c;就是为了理想的生活 ! 目录 ⭐ 专栏简介 &#x1f4d8; 文章引言 一、H…

CentOS 搭建 Hadoop3 高可用集群

Hadoop FullyDistributed Mode 完全分布式 spark101spark102spark103192.168.171.101192.168.171.102192.168.171.103namenodenamenodejournalnodejournalnodejournalnodedatanodedatanodedatanodenodemanagernodemanagernodemanagerrecource managerrecource managerjob hist…

2023.10.31 关于 Spring 的基本概念

目录 Spring 容器 对象生命周期 IoC Spring IoC DI Spring Spring 的全称为 Spring Framework&#xff0c;是一个开源的 Java 应用程序框架它提供了一种综合的编程和配置模型&#xff0c;用于构建现代化企业级的应用程序 一句话概括 Spring 是包含了众多工具方法的 IoC …

docker进阶

文章目录 docker 进阶Part1 常用命令总结docker version 查看docker客户端和服务端信息docker info 查看更加详细信息docker images 列出所有镜像基本用法常用选项 docker search 搜索镜像基本用法示例用法 docker pull 拉取镜像基本用法示例用法 docker rmi 删除镜像基本用法示…

FMC子卡解决方案:FMC214-基于FMC兼容1.8V IO的Full Camera Link 输出子卡

FMC214-基于FMC兼容1.8V IO的Full Camera Link 输出子卡 一、板卡概述   基于FMC兼容1.8V IO的Full Camera Link 输出子卡支持Base、Middle、Full Camera link信号输出&#xff0c;兼容1.8V、2.5V、3.3V IO FPGA信号输出。适配xilinx不同型号开发板和公司内部各FMC载板。北…

Macroscope安全漏洞检测工具简介

学习目标&#xff1a; 本介绍旨在帮助感兴趣者尽快了解 Macroscope&#xff0c;这是一款用于安全测试自动化和漏洞管理的企业工具。 全覆盖应用程序安全测试&#xff1a; 如下图所示&#xff0c;如果使用多种互补工具&#xff08;SAST/DAST/SCA 等&#xff09;来检测应用程序…

webgoat(A2) Broken Authentication

身份验证绕过 身份验证绕过以多种方式发生&#xff0c;但通常会利用配置或逻辑中的某些缺陷。篡改以达到正确的条件。 隐藏输入 最简单的形式是依赖于网页/DOM 中的隐藏输入。 删除参数 有时&#xff0c;如果攻击者不知道参数的正确值&#xff0c;他们可能会从提交中完全删…

安企神局域网监控软件,员工电脑终端的安全管理软件

安企神局域网监控软件&#xff0c;员工电脑终端的安全管理软件 安企神局域网监控软件下载使用 公司老板其实最怕的就是公司机密遭到泄露&#xff0c;而一般泄露的方法都是通过一些通讯软件而泄露出去的&#xff0c;如微信、qq等等&#xff0c;所以很多老板都想知道有什么软件…

idea 配置checkstyle全过程

checkstyle是提高代码质量,检查代码规范的很好用的一款工具&#xff0c;本文简单介绍一下集成的步骤&#xff0c;并提供一份完整的checkstyle的代码规范格式文件&#xff0c;以及常见的格式问题的解决方法。 一&#xff0c;安装 打开idea的文件选项&#xff0c;选择设置&…

Intel oneAPI笔记(1)--oneAPI简介、SYCL编程简介

oneAPI简介 Intel oneAPI是Intel提供的统一编程模型和软件开发框架。 它旨在简化可充分利用英特尔各种硬件架构&#xff08;包括 CPU、GPU 和 FPGA&#xff09;的应用程序的开发 oneAPI一个重要的特性是开放性&#xff0c;支持多种类型的架构和不同的硬件供应商&#xff0c;是…

记录一次时序数据库的实战测试

0x1.前言 本文章仅用于信息安全防御技术分享&#xff0c;因用于其他用途而产生不良后果&#xff0c;作者不承担任何法律责任&#xff0c;请严格遵循中华人民共和国相关法律法规&#xff0c;禁止做一切违法犯罪行为。文中涉及漏洞均以提交至教育漏洞平台。 0x2.背景 在某次Edus…

【Redis】Redis常用命令-getsetkeysexistsexpirettltype

文章目录 读取文档注意事项set命令get命令全局/通用命令KEYSEXISTSDELEXPIRETTLTYPE 读取文档注意事项 官方文档链接&#xff1a;https://redis.io/ 注意&#xff1a;redis的命令不区分大小写 在redis文档给出的语法格式说明&#xff1a; []&#xff1a;相当于一个独立的单元&a…

golang 发起 http 请求,获取访问域名的 ip 地址(net, httptrace)

前言 今天碰到了个需求&#xff0c;我要知道程序对外访问的 http 请求域名的 ip 地址。 直接查看 golang 的 net/http 包&#xff0c;发现 Response 中并没有我想要的 ip 信息。 考虑到在 OSI 七层模型中&#xff0c;ip 是网络层协议&#xff0c;而 http 是应用层协议。去翻…