[C++]AVL树怎么转

AVL树是啥

一提到AVL树,脑子里不是旋了,就是悬了。

AVL树之所以难,并不是因为结构难以理解,而是因为他的旋转。

AVL树定义

  • 平衡因子:对于一颗二叉树,某节点的左右子树高度之差,就是该节点的平衡因子
  • AVL树:对于一颗二叉树,任意节点的平衡因子bf 在范围[-1,1]之间(即左右子树高度差的绝对值<=1),则该树就是平衡二叉树

为何会有AVL树

AVL树是二叉搜索树的衍生,其名字来源是根据两位俄罗斯的数学家G.M.Adelson-VelskiiE.M.Landis,他们在1962年发明的一种用来解决二叉搜索树在极端情况下时间复杂度变为O(n)的情况。而其解决该情况的方法便是:通过旋转旋转来调整二叉搜索树的平衡

AVL树的节点

AVL树的节点实现方式有很多,这里采取下面的方法定义节点:

template<class T>struct AVLTreeNode{AVLTreeNode(const T& data): _left(nullptr),_right(nullptr),_parent(nullptr),_data(data),_bf(0){}AVLTreeNode<T>* _left;  // 该节点的左孩子AVLTreeNode<T>* _right; // 该节点的右孩子AVLTreeNode<T>* _parent;// 该节点的父亲T _data;//存储数据int _bf;//平衡因子  
};

这里AVL树节点的实现增加了_bf (平衡因子)成员变量,用来记录每个点的平衡因子。同时用了父节点的指针_parent ,目的是方便调整平衡因子。

AVL树的插入

这里AVL树的插入操作与二叉搜索树一样,唯一不同的是,在插入后要调整平衡因子

  • 对于插入的节点,因为其是插入到叶节点位置,所以他的平衡因子为0
  • 将节点插入后,树的高度可能会发生改变,此时则要调整他父亲,甚至还要调整父亲的父亲的平衡因子。主要分为以下几种情况

情况1:

在这里插入图片描述

如果在节点的右孩子插入,则该节点的bf需要+1(节点70、50的bf连锁着发生了变化,后面有说明)


情况2:

在这里插入图片描述

如果在节点的左孩子插入,则该节点的bf需要-1(节点70、50的bf连锁着发生了变化,后面有说明)

但是没完!

如果节点的平衡因子因为插入而变成了1 或者-1 ,则说明子树的高度发生了变化,此时该节点的父节点bf也应发生变化(如果父节点的bf更改之后影响了“爷爷节点”,则爷爷节点也要跟着跟着变化)。所以上图中的70和50两节点的bf也要发生变化。


但是还没完!

bf 的值有可能会变为2、-2,则此时就需要进行旋转操作(旋转完成后,会手动调整bf,保证其符合要求)

代码:

//插入操作
...
//调整平衡因子
cur = newnode;
parent = newnode->_parent;
while (parent)
{if (parent->_right == cur){++parent->_bf;}else if (parent->_left == cur){--parent->_bf;}if (parent->_bf == 0)//AVL树稳定了,break出去break;else if (parent->_bf == 1 || parent->_bf == -1)//无需旋转但是需要更改父亲的平衡因子{cur = parent;parent = parent->_parent;}else if ()//需要旋转{}
}

AVL树的旋转

重中之重,也是难中之难

AVL树旋转的目的

AVL树是一个平衡二叉搜索树,因为某些插入操作导致它不再平衡(具体表现是平衡因子变为2或-2),则此时为了使之平衡,就需要进行旋转操作

AVL树旋转操作

AVL树的旋转主要分为4种情况:

  1. 左旋
  2. 右旋
  3. 左旋再右旋(双旋)
  4. 右旋再左旋(双旋)

情况1:左旋

在这里插入图片描述

对于插入后父亲的bf=2,右孩子的bf=1的情况,采用左旋。且左旋后平衡因子都是0

在这里插入图片描述


情况2:右旋

在这里插入图片描述

对于插入后父亲的bf=-2,左孩子的bf=-1的情况,采用右旋。且右旋后平衡因子都是0

在这里插入图片描述


双旋

这种情况的发生是因为发现对该节点无论左旋还是右旋,都无法使其平衡,原因是插入节点的位置比较特殊

情况3:左旋再右旋

在这里插入图片描述

对于插入(这里插入b还是c只会影响旋转完后的平衡因子)后父亲的bf=-2,左孩子的bf=1的情况,采用左旋再右旋。这里的平衡因子更新规则与前不同,详见后文。

在这里插入图片描述


情况4:先右旋再左旋

在这里插入图片描述

对于插入(这里插入b还是c只会影响旋转完后的平衡因子)后父亲的bf=2,右孩子的bf=-1的情况,采用右旋再左旋。这里的平衡因子更新规则与前不同,详见后文。
在这里插入图片描述

双旋的平衡因子更新规则

更新规则相比旋转规则就简单许多

分为以下3种情况(以上图为例):

  1. 如果60的平衡因子(旋转前)是-1,则更新后的60的平衡因子变为0,左孩子变为0,右孩子变为1
  2. 如果60的平衡因子(旋转前)是1,则更新后60的平衡因子变为0,左孩子变为-1,右孩子变为0
  3. 如果60的平衡因子(旋转前)是0,则更新后60的平衡因子变为0,左孩子变为0,右孩子变为0

代码:

这里只给出一部分代码,剩下的可以自己尝试写出来,然后可以到仓库对照

bool insert(const pair<K,V>& kv)
{//插入Node* newnode = new Node(kv);if (newnode == nullptr) return false;if (_root == nullptr){_root = newnode;return true;}Node* cur = _root;Node* parent = nullptr;while (cur){parent = cur;if (kv.first > cur->_kv.first){cur = cur->_right;}else cur = cur->_left;}//调整节点连接if (kv.first > parent->_kv.first)parent->_right = newnode;else parent->_left = newnode;newnode->_parent = parent;//调整平衡因子cur = newnode;parent = newnode->_parent;while (parent){if (parent->_right == cur){++parent->_bf;}else if (parent->_left == cur){--parent->_bf;}if (parent->_bf == 0)//AVL树稳定了,break出去break;else if (parent->_bf == 1 || parent->_bf == -1)//无需旋转但是需要更改父亲的平衡因子{cur = parent;parent = parent->_parent;}else if (parent->_bf == 2 || parent->_bf == -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){RotateRL(parent);}else if (parent->_bf == -2 && cur->_bf == 1){RotateLR(parent);}else{assert(false);}}}return true;
}
//左旋
void RotateL(Node* parent)
{Node* subR = parent->_right;Node* subRL = subR->_left;Node* pparent = parent->_parent;parent->_parent = subR;subR->_left = parent;parent->_right = subRL;if (subRL){subRL->_parent = parent;}if (!pparent){_root = subR;subR->_parent = nullptr;}else{if (pparent->_left == parent){pparent->_left = subR;}else{pparent->_right = subR;}subR->_parent = pparent;}parent->_bf = 0;subR->_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 == 0){subLR->_bf = 0;parent->_bf = 0;subL->_bf = 0;}else if (bf == 1){subLR->_bf = 0;parent->_bf = 0;subL->_bf = -1;}else if (bf == -1){subLR->_bf = 0;parent->_bf = 1;subL->_bf = 0;}else{assert(false);}
}

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

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

相关文章

第三百七十八回

文章目录 1. 概念介绍2. 实现方法2.1 maskFilter2.2 shader 3. 代码与效果3.1 示例代码3.2 运行效果 4. 内容总结 我们在上一章回中介绍了"两种阴影效果"相关的内容&#xff0c;本章回中将介绍如何绘制阴影效果.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概…

【C++庖丁解牛】类与对象

&#x1f4d9; 作者简介 &#xff1a;RO-BERRY &#x1f4d7; 学习方向&#xff1a;致力于C、C、数据结构、TCP/IP、数据库等等一系列知识 &#x1f4d2; 日后方向 : 偏向于CPP开发以及大数据方向&#xff0c;欢迎各位关注&#xff0c;谢谢各位的支持 目录 1.面向过程和面向对象…

构建大语言模型的四个主要阶段

大规模语言模型的发展历程虽然只有短短不到五年的时间&#xff0c;但是发展速度相当惊人&#xff0c;国内外有超过百种大模型相继发布。中国人民大学赵鑫教授团队在文献按照时间线给出 2019 年至 2023 年比较有影响力并且模型参数量超过 100 亿的大规模语言模型。大规模语言模型…

基于雷达影像的洪水监测技术方法详解

洪水发生时候大多数是阴雨天气&#xff0c;光学影像基本上拍不到有效影像。雷达影像这时候就能发挥其不受天气影像的优点。现在星载的雷达卫星非常多&#xff0c;如高分三号、陆探一号、海丝一号&#xff08;巢湖一号&#xff09;、哨兵1号等。本文以哨兵1号L1地距(GRD)产品来介…

【初中生讲机器学习】13. 决策树算法一万字详解!一篇带你看懂!

创建时间&#xff1a;2024-03-02 最后编辑时间&#xff1a;2024-03-02 作者&#xff1a;Geeker_LStar 你好呀~这里是 Geeker_LStar 的人工智能学习专栏&#xff0c;很高兴遇见你~ 我是 Geeker_LStar&#xff0c;一名初三学生&#xff0c;热爱计算机和数学&#xff0c;我们一起加…

电子电器架构 —— DoIP协议相关的介绍

电子电器架构 —— DoIP协议相关的介绍 我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 没有人关注你。也无需有人关注你。你必须承认自己的价值,你不能站在他人的角度来反对自己。人生在世,最怕…

Excel 按奇数偶数列处理数据

目录 一. 需求背景1.1 获取偶数列的数据1.2 奇偶列数据互换 二. 解决方式2.1 为列添加奇偶辅助列2.2 通过公式将奇偶列互换 一. 需求背景 1.1 获取偶数列的数据 ⏹ 最近在整理歌单&#xff0c;发现部分歌曲没有歌词&#xff0c;于是打算自己制作一份。 从网上找到了歌词&…

【Spring IoC】实验四:特殊值处理

个人名片&#xff1a; &#x1f43c;作者简介&#xff1a;一名大三在校生&#xff0c;喜欢AI编程&#x1f38b; &#x1f43b;‍❄️个人主页&#x1f947;&#xff1a;落798. &#x1f43c;个人WeChat&#xff1a;hmmwx53 &#x1f54a;️系列专栏&#xff1a;&#x1f5bc;️…

Topaz Gigapixel AI:让每一张照片都焕发新生mac/win版

Topaz Gigapixel AI 是一款革命性的图像增强软件&#xff0c;它利用先进的人工智能技术&#xff0c;能够显著提升图像的分辨率和质量。无论是摄影爱好者还是专业摄影师&#xff0c;这款软件都能帮助他们将模糊的、低分辨率的照片转化为清晰、细腻的高分辨率图像。 Topaz Gigap…

《Spring Security 简易速速上手小册》第9章 测试与维护 (2024 最新版)

文章目录 9.1 编写安全测试9.1.1 基础知识9.1.2 重点案例&#xff1a;保护 REST API9.1.3 拓展案例 1&#xff1a;自定义登录逻辑测试9.1.4 拓展案例 2&#xff1a;CSRF 保护测试 9.2 Spring Security 升级和维护9.2.1 基础知识9.2.2 重点案例&#xff1a;适配新的密码存储格式…

监控与日志

一、监控 1、监控类型 从监控类型上划分&#xff0c;在 K8s 中可以分成四个不同的类型&#xff1a; ① 资源监控&#xff1a;这种监控主要关注于基础资源的使用情况&#xff0c;例如 CPU、内存、网络等。通常使用数值或百分比等单位来统计&#xff0c;可以通过 Zabbix、Tele…

vue3项目中如何一个vue组件中的一个div里面的图片铺满整个屏幕样式如何设置

在Vue 3项目中&#xff0c;要使一个div内的图片铺满整个屏幕&#xff0c;你需要确保几个关键点&#xff1a;div元素和图片元素的样式设置正确&#xff0c;以及确保它们能够覆盖整个视口&#xff08;viewport&#xff09;。以下是一个简单的步骤和代码示例&#xff0c;帮助你实现…