平衡二叉树
(1)由来:平衡二叉树是基于二分法的策略提高数据的查找速度的二叉树的数据结构;
(2)特点:
平衡二叉树是采用二分法思维把数据按规则组装成一个树形结构的数据,用这个树形结构的数据减少无关数据的检索,
大大的提升了数据检索的速度;平衡二叉树的数据结构有以下规则:
1.非叶子节点只能允许最多两个子节点的存在
2.每一个非叶子节点数据分布规则为左边的子节点小当前节点的值,
3.右边的子节点大于当前节点的值(这里值是基于自己的算法规则而定的,比如hash值);
平衡树的层级结构:
因为平衡二叉树查询性能和树的层级(h高度)成正比、为了保证树的结构左右两端数据大致平衡降低二叉树的查询难度一般会采用一种算法机制实现节点数据结构的平衡
实现了这种算法的有比如AVL、Treap、红黑树,使用平衡二叉树能保证数据的左右两边的节点层级相差不会大于 1
通过这样避免树形结构由于删除增加变成线性链表影响查询效率,保证数据平衡的情况下查找数据的速度近于二分法查找;
平衡二叉树特点:
1.非叶子节点最多拥有两个子节点;
2.左边的子节点小当前节点的值
3.右边的子节点大于当前节点的值
4.树的左右两边的层级数相差不会大于 1
5.没有值相等重复的节点;
6.具有二叉查找树的全部特性。
二叉树的优点:二叉树详细步骤
1.二叉排序树是一种比较有用的折衷方案。
2.数组的搜索比较方便,可以直接用下标,但删除或者插入某些元素就比较麻烦。
3.链表与之相反,删除和插入元素很快,但查找很慢。
4.二叉排序树就既有链表的好处,也有数组的好处。
5.在处理大批量的动态的数据是比较有用。
平衡二叉树优点:
1.平衡二叉树伸展树(SplayTree)是一种二叉排序树,它能在O(logn)内完成插入、查找和删除操作
2.平衡二叉树主要优点集中在快速查找
二叉树插入如下图解:
图一就是一颗AVL树了,而图二则不是(节点右边标的是这个节点的高度)。
![089703007a2ef93a1931b2f0b435bf04.png](https://img-blog.csdnimg.cn/img_convert/089703007a2ef93a1931b2f0b435bf04.png)
![2349c0aa8e6075eb951dd54ba962575f.png](https://img-blog.csdnimg.cn/img_convert/2349c0aa8e6075eb951dd54ba962575f.png)
对于图二,因为节点9的左孩子高度为2,而右孩子高度为0。他们之间的差值超过1了。
这种树就可以保证不会出现大量节点偏向于一边的情况了。
当我们在进行节点插入的时候,可能会出现节点都倾向于左边的情况,例如:
![70007023160b02abd4583221807877ce.png](https://img-blog.csdnimg.cn/img_convert/70007023160b02abd4583221807877ce.png)
我们把这种倾向于左边的情况称之为 左-左型。这个时候,我们就可以对节点 9 进行右旋操作,使它恢复平衡。
![2b3dc9fa5f73fcf9f2edc69d05de0d19.png](https://img-blog.csdnimg.cn/img_convert/2b3dc9fa5f73fcf9f2edc69d05de0d19.png)
即:顺时针旋转两个节点,使得父节点被自己的左孩子取代,而自己成为自己的右孩子
再举个例子:
![f65f9883620e6dd70c4ba0048852054c.png](https://img-blog.csdnimg.cn/img_convert/f65f9883620e6dd70c4ba0048852054c.png)
节点4和9高度相差大于1。由于是左孩子的高度较高,此时是左-左型,进行右旋。
![59e4f8088ebd138298f5bb9078435a2d.png](https://img-blog.csdnimg.cn/img_convert/59e4f8088ebd138298f5bb9078435a2d.png)
这里要注意,节点4的右孩子成为了节点6的左孩子了
![04cd747bff797fdd5e335a68cd218666.gif](https://img-blog.csdnimg.cn/img_convert/04cd747bff797fdd5e335a68cd218666.gif)
左旋
左旋和右旋一样,就是用来解决当大部分节点都偏向右边的时候,通过左旋来还原。例如:
![0e7e26b6282b7c7747ceb3e7569f4071.png](https://img-blog.csdnimg.cn/img_convert/0e7e26b6282b7c7747ceb3e7569f4071.png)
我们把这种倾向于右边的情况称之为 右-右型。
![823f3538e12cfb62de0f62e916d0b894.gif](https://img-blog.csdnimg.cn/img_convert/823f3538e12cfb62de0f62e916d0b894.gif)
例子讲解:初始状态如下:
![fed275b237e5fcee8e616516cf9bb57b.png](https://img-blog.csdnimg.cn/img_convert/fed275b237e5fcee8e616516cf9bb57b.png)
然后我们主键插入如下数值:1,4,5,6,7,10,9,8
插入 1
![53f3d09f0a701cafcfae34a128474dd4.png](https://img-blog.csdnimg.cn/img_convert/53f3d09f0a701cafcfae34a128474dd4.png)
左-左型,需要右旋调整。
![00c4ff677f8fa293a1366d044c0b4108.png](https://img-blog.csdnimg.cn/img_convert/00c4ff677f8fa293a1366d044c0b4108.png)
插入4
![1a3071fed1305d2ec8cfb158e83b5a67.png](https://img-blog.csdnimg.cn/img_convert/1a3071fed1305d2ec8cfb158e83b5a67.png)
继续插入 5
![760e7b495a525ee57488d252cc90663e.png](https://img-blog.csdnimg.cn/img_convert/760e7b495a525ee57488d252cc90663e.png)
右-右型,需要左旋转调整。
![4160c556ff8b3d9563ca74ecb566e51e.png](https://img-blog.csdnimg.cn/img_convert/4160c556ff8b3d9563ca74ecb566e51e.png)
继续插入6
![ffa7fe1871a1bff4c6c761de8f9e9d8a.png](https://img-blog.csdnimg.cn/img_convert/ffa7fe1871a1bff4c6c761de8f9e9d8a.png)
右-右型,需要进行左旋
![03c629a5cf9bb1bd66894742e192da5c.png](https://img-blog.csdnimg.cn/img_convert/03c629a5cf9bb1bd66894742e192da5c.png)
继续插入7
![82f4aaebdf59c422fd369bb528b8e1ae.png](https://img-blog.csdnimg.cn/img_convert/82f4aaebdf59c422fd369bb528b8e1ae.png)
右-右型,需要进行左旋
![a87ab1e21e37f3a115ce5e97516ee927.png](https://img-blog.csdnimg.cn/img_convert/a87ab1e21e37f3a115ce5e97516ee927.png)
继续插入10
![f27bcbfcb7bfb8bf5227cbcd73609726.png](https://img-blog.csdnimg.cn/img_convert/f27bcbfcb7bfb8bf5227cbcd73609726.png)
继续插入9
![3fb7a200810493eed7343fc3531191ee.png](https://img-blog.csdnimg.cn/img_convert/3fb7a200810493eed7343fc3531191ee.png)
出现了这种情况怎么办呢?对于这种 右-左型 的情况,单单一次左旋或右旋是不行的,下面我们先说说如何处理这种情况,例如:
![d5663a5499b61f7c14a874f6a7592a38.png](https://img-blog.csdnimg.cn/img_convert/d5663a5499b61f7c14a874f6a7592a38.png)
这种我们就把它称之为 右-左 型吧。处理的方法是先对节点10进行右旋把它变成右-右型。
![649a8f7c08655b190445b59daf0d6a0e.png](https://img-blog.csdnimg.cn/img_convert/649a8f7c08655b190445b59daf0d6a0e.png)
然后在进行左旋。
![dcc3dfe50ca03b38ebecf0016141106d.png](https://img-blog.csdnimg.cn/img_convert/dcc3dfe50ca03b38ebecf0016141106d.png)
所以对于这种 右-左型的,我们需要进行一次右旋再左旋。
同理,也存在 左-右型的,例如:
![973bd0d5fd2d73d34b20d68c00d12724.png](https://img-blog.csdnimg.cn/img_convert/973bd0d5fd2d73d34b20d68c00d12724.png)
对于左-右型的情况和刚才的 右-左型相反,我们需要对它进行一次左旋,再右旋。
![ed467d35c848a21ffbf700d1a0690021.png](https://img-blog.csdnimg.cn/img_convert/ed467d35c848a21ffbf700d1a0690021.png)
回到刚才那道题
![0991217f48d171683c0bc6fb557fcce7.png](https://img-blog.csdnimg.cn/img_convert/0991217f48d171683c0bc6fb557fcce7.png)
对它进行右旋再左旋。
![d4599eb5755b6a93862cd355b03b3ba4.png](https://img-blog.csdnimg.cn/img_convert/d4599eb5755b6a93862cd355b03b3ba4.png)
到此,我们的插入就结束了。
失去平衡的调整方法:
- 解决失衡的方法口诀
- 左左失衡,parent右旋;
- 左右失衡,cur左旋,parent右旋;
- 右右失衡,parent左旋;
- 右左失衡,cur右旋,parent左旋;
解析:parent为当前失衡的结点,cur为parent的孩子(新结点所在的子树的根节点)
![f1bfe7abebe907ab4da1efbd1b1d3e45.png](https://img-blog.csdnimg.cn/img_convert/f1bfe7abebe907ab4da1efbd1b1d3e45.png)
-
- 四种失衡状态:
![b0b9db793753dab6b4404ca34c6ca206.png](https://img-blog.csdnimg.cn/img_convert/b0b9db793753dab6b4404ca34c6ca206.png)
![2342c2e2c7d45c3dd1ce498f07df0fa3.png](https://img-blog.csdnimg.cn/img_convert/2342c2e2c7d45c3dd1ce498f07df0fa3.png)
-
- 很明显:如果从根节点到新插入的结点是一路向左,即左左失衡;
- 如果中途向右了,即左右失衡;
- 如果一路向右,即为右右失衡;
- 如果中途向左了,即右左失衡;
- 根据不同的失衡,口诀中有不同的解决方法,接下来我们一起来验证一下吧:
- ”左左失衡,parent右旋“:
![e91ffd4fb2986d2d1b4e1c5bb8b71cbb.png](https://img-blog.csdnimg.cn/img_convert/e91ffd4fb2986d2d1b4e1c5bb8b71cbb.png)
-
- “左右失衡,cur左旋,parent右旋”:
![8bf07e68542a45078d630a35c329437d.png](https://img-blog.csdnimg.cn/img_convert/8bf07e68542a45078d630a35c329437d.png)
-
- “右右失衡,parent左旋”:
![de6b2a8963a0e93e48bd1f96a2292f74.png](https://img-blog.csdnimg.cn/img_convert/de6b2a8963a0e93e48bd1f96a2292f74.png)
-
-
- “右左失衡,cur右旋,parent左旋”:
-
![8cab84e8ee67ebcc0ae1a35aae8ff49d.png](https://img-blog.csdnimg.cn/img_convert/8cab84e8ee67ebcc0ae1a35aae8ff49d.png)
- 调整之后,该树还是一个平衡二叉树
部分转载链接:https://blog.csdn.net/weixin_42125310/article/details/105183623