当前位置: 首页 > article >正文

【数据结构与算法】第28篇:平衡二叉树(AVL树)

一、AVL树的定义1.1 平衡因子平衡因子 左子树高度 - 右子树高度AVL树要求所有节点的平衡因子只能是 -1、0、1。text节点高度从该节点到最远叶子节点的边数 空树高度-1 或 0不同定义本文用-11.2 为什么需要平衡普通BST插入有序序列时会退化text插入1,2,3,4,5 退化后 1 \ 2 \ 3 \ 4 \ 5 查找效率 O(n)AVL树通过旋转保持平衡查找效率始终 O(log n)。二、节点结构ctypedef struct AVLNode { int data; struct AVLNode *left; struct AVLNode *right; int height; // 节点高度 } AVLNode, *AVLTree;2.1 辅助函数c// 获取节点高度 int getHeight(AVLNode *node) { return node NULL ? -1 : node-height; } // 计算平衡因子 int getBalanceFactor(AVLNode *node) { if (node NULL) return 0; return getHeight(node-left) - getHeight(node-right); } // 更新节点高度 void updateHeight(AVLNode *node) { if (node NULL) return; int leftH getHeight(node-left); int rightH getHeight(node-right); node-height (leftH rightH ? leftH : rightH) 1; }三、四种旋转调整3.1 右旋LL型触发条件平衡因子 1 且 左子树的平衡因子 ≥ 0text不平衡节点: A 左孩子: B A B / \ / \ B C 右旋→ D A / \ / \ D E E CcAVLNode* rightRotate(AVLNode *y) { AVLNode *x y-left; AVLNode *T2 x-right; // 旋转 x-right y; y-left T2; // 更新高度 updateHeight(y); updateHeight(x); return x; }3.2 左旋RR型触发条件平衡因子 -1 且 右子树的平衡因子 ≤ 0text不平衡节点: A 右孩子: B A B / \ / \ C B 左旋→ A E / \ / \ D E C DcAVLNode* leftRotate(AVLNode *y) { AVLNode *x y-right; AVLNode *T2 x-left; // 旋转 x-left y; y-right T2; // 更新高度 updateHeight(y); updateHeight(x); return x; }3.3 左右旋LR型触发条件平衡因子 1 且 左子树的平衡因子 0text步骤1对左孩子左旋 步骤2对当前节点右旋 A A C / \ / \ / \ B C 左旋→ C C 右旋→ B A / \ / / \ \ D E B D E C / \ D EcAVLNode* leftRightRotate(AVLNode *node) { node-left leftRotate(node-left); return rightRotate(node); }3.4 右左旋RL型触发条件平衡因子 -1 且 右子树的平衡因子 0text步骤1对右孩子右旋 步骤2对当前节点左旋cAVLNode* rightLeftRotate(AVLNode *node) { node-right rightRotate(node-right); return leftRotate(node); }四、插入操作cAVLNode* insert(AVLNode *root, int value) { // 1. 普通BST插入 if (root NULL) { AVLNode *newNode (AVLNode*)malloc(sizeof(AVLNode)); newNode-data value; newNode-left NULL; newNode-right NULL; newNode-height 0; return newNode; } if (value root-data) { root-left insert(root-left, value); } else if (value root-data) { root-right insert(root-right, value); } else { return root; // 重复值不插入 } // 2. 更新高度 updateHeight(root); // 3. 计算平衡因子判断是否需要旋转 int balance getBalanceFactor(root); // 左子树过高 if (balance 1) { if (value root-left-data) { // LL型 return rightRotate(root); } else { // LR型 return leftRightRotate(root); } } // 右子树过高 if (balance -1) { if (value root-right-data) { // RR型 return leftRotate(root); } else { // RL型 return rightLeftRotate(root); } } return root; }五、完整代码演示c#include stdio.h #include stdlib.h typedef struct AVLNode { int data; struct AVLNode *left; struct AVLNode *right; int height; } AVLNode, *AVLTree; int getHeight(AVLNode *node) { return node NULL ? -1 : node-height; } int getBalanceFactor(AVLNode *node) { if (node NULL) return 0; return getHeight(node-left) - getHeight(node-right); } void updateHeight(AVLNode *node) { if (node NULL) return; int leftH getHeight(node-left); int rightH getHeight(node-right); node-height (leftH rightH ? leftH : rightH) 1; } AVLNode* rightRotate(AVLNode *y) { AVLNode *x y-left; AVLNode *T2 x-right; x-right y; y-left T2; updateHeight(y); updateHeight(x); return x; } AVLNode* leftRotate(AVLNode *y) { AVLNode *x y-right; AVLNode *T2 x-left; x-left y; y-right T2; updateHeight(y); updateHeight(x); return x; } AVLNode* leftRightRotate(AVLNode *node) { node-left leftRotate(node-left); return rightRotate(node); } AVLNode* rightLeftRotate(AVLNode *node) { node-right rightRotate(node-right); return leftRotate(node); } AVLNode* insert(AVLNode *root, int value) { if (root NULL) { AVLNode *newNode (AVLNode*)malloc(sizeof(AVLNode)); newNode-data value; newNode-left NULL; newNode-right NULL; newNode-height 0; return newNode; } if (value root-data) { root-left insert(root-left, value); } else if (value root-data) { root-right insert(root-right, value); } else { return root; } updateHeight(root); int balance getBalanceFactor(root); // LL if (balance 1 value root-left-data) { return rightRotate(root); } // RR if (balance -1 value root-right-data) { return leftRotate(root); } // LR if (balance 1 value root-left-data) { return leftRightRotate(root); } // RL if (balance -1 value root-right-data) { return rightLeftRotate(root); } return root; } void inorder(AVLNode *root) { if (root NULL) return; inorder(root-left); printf(%d , root-data); inorder(root-right); } void preorder(AVLNode *root) { if (root NULL) return; printf(%d , root-data); preorder(root-left); preorder(root-right); } void printTree(AVLNode *root, int level) { if (root NULL) return; printTree(root-right, level 1); for (int i 0; i level; i) printf( ); printf(%d(h%d,bf%d)\n, root-data, root-height, getBalanceFactor(root)); printTree(root-left, level 1); } int main() { AVLNode *root NULL; printf( 插入序列: 10, 20, 30, 40, 50, 25 \n); int values[] {10, 20, 30, 40, 50, 25}; for (int i 0; i 6; i) { root insert(root, values[i]); printf(\n插入 %d 后:\n, values[i]); printf(中序遍历: ); inorder(root); printf(\n); printf(树结构:\n); printTree(root, 0); } return 0; }运行结果text 插入序列: 10, 20, 30, 40, 50, 25 插入 10 后: 中序遍历: 10 树结构: 10(h0,bf0) 插入 20 后: 中序遍历: 10 20 树结构: 20(h1,bf-1) 10(h0,bf0) 插入 30 后: 中序遍历: 10 20 30 树结构: 30(h1,bf-1) 20(h0,bf0) 10(h0,bf0) 插入 40 后: 中序遍历: 10 20 30 40 树结构: 40(h2,bf-2) 30(h1,bf-1) 20(h0,bf0) 10(h0,bf0) 插入 50 后: 中序遍历: 10 20 30 40 50 树结构: 50(h2,bf-2) 40(h1,bf-1) 30(h0,bf0) 20(h0,bf0) 10(h0,bf0) 插入 25 后: 中序遍历: 10 20 25 30 40 50 树结构: 50(h2,bf-1) 40(h1,bf1) 30(h0,bf0) 25(h0,bf0) 20(h1,bf0) 10(h0,bf0)六、旋转判断总结平衡因子子树平衡因子类型操作1≥0LL右旋10LR左右旋-1≤0RR左旋-10RL右左旋记忆口诀LL左边太重右旋RR右边太重左旋LR左子树的右边太重先左旋左子再右旋RL右子树的左边太重先右旋右子再左旋七、复杂度分析操作时间复杂度说明查找O(log n)树高始终 log n插入O(log n)查找位置 旋转常数次删除O(log n)类似插入AVL树的树高严格控制在 log n 级别性能稳定。八、AVL树 vs 普通BST对比项普通BSTAVL树最坏查找O(n)O(log n)插入复杂度O(log n)平均O(log n)实现复杂度简单复杂需要旋转额外开销无每个节点存储高度适用场景数据随机数据有序或对性能要求高九、小结这一篇我们学习了AVL树要点说明平衡因子左高-右高取值-1,0,1LL右旋RR左旋LR左旋左子 右旋RL右旋右子 左旋插入BST插入 更新高度 旋转平衡核心思想在BST插入后从插入点向上检查平衡因子发现不平衡就旋转调整。下一篇我们讲红黑树原理与C语言模拟。十、思考题为什么AVL树的高度平衡能保证查找效率O(log n)在LR旋转中为什么需要先左旋再右旋直接右旋行不行如果插入序列是30,20,10会触发哪种旋转画图说明。尝试实现AVL树的删除操作。欢迎在评论区讨论你的答案。

相关文章:

【数据结构与算法】第28篇:平衡二叉树(AVL树)

一、AVL树的定义1.1 平衡因子平衡因子 左子树高度 - 右子树高度AVL树要求所有节点的平衡因子只能是 -1、0、1。text节点高度:从该节点到最远叶子节点的边数 空树高度:-1 或 0(不同定义,本文用-1)1.2 为什么需要平衡普…...

【数据结构与算法】第27篇:二叉排序树(BST

一、二叉排序树的定义1.1 性质二叉排序树&#xff08;Binary Search Tree&#xff0c;BST&#xff09;满足以下性质&#xff1a;左子树所有节点的值 < 根节点的值右子树所有节点的值 > 根节点的值左右子树本身也是二叉排序树示例&#xff1a;text50/ \30 70/ \ / \2…...

obsidian-skills培训管理:培训用户使用技能的方法

obsidian-skills培训管理&#xff1a;培训用户使用技能的方法 【免费下载链接】obsidian-skills Agent skills for Obsidian. Teach your agent to use Markdown, Bases, JSON Canvas, and use the CLI. 项目地址: https://gitcode.com/GitHub_Trending/ob/obsidian-skills …...

终极指南:php-webdriver弹窗处理与WebDriverAlert对话框管理技巧

终极指南&#xff1a;php-webdriver弹窗处理与WebDriverAlert对话框管理技巧 【免费下载链接】php-webdriver PHP client for Selenium/WebDriver protocol. Previously facebook/php-webdriver 项目地址: https://gitcode.com/gh_mirrors/ph/php-webdriver 想要掌握PHP…...

K3s证书过期急救指南:5分钟搞定证书轮换(附一键脚本)

K3s证书过期急救指南&#xff1a;5分钟搞定证书轮换&#xff08;附一键脚本&#xff09; 凌晨三点&#xff0c;报警短信突然炸响——K3s集群所有服务不可用。登录控制台看到满屏的x509: certificate has expired or is not yet valid报错时&#xff0c;我才意识到证书过期这个&…...

保姆级教程:用Keil5将你的STM32F103工程无缝迁移到国民技术N32G45X

从STM32F103到N32G45X&#xff1a;嵌入式工程师的国产MCU迁移实战指南 在嵌入式开发领域&#xff0c;芯片选型往往决定着项目的成败。随着国产微控制器的崛起&#xff0c;越来越多的工程师开始考虑将原有基于STM32的项目迁移到国产平台。国民技术的N32G45X系列以其出色的性价比…...

正则表达式元字符详解:learn-regex-zh 进阶教程

正则表达式元字符详解&#xff1a;learn-regex-zh 进阶教程 【免费下载链接】learn-regex-zh :cn: 翻译: 学习正则表达式的简单方法 项目地址: https://gitcode.com/gh_mirrors/le/learn-regex-zh 正则表达式是一种强大的文本处理工具&#xff0c;而元字符是构建正则表达…...

10点滑动平均滤波器:嵌入式零依赖高效实现

1. 项目概述MovingAverageFilter 是一个轻量级、零依赖的嵌入式数字滤波器实现&#xff0c;专为资源受限的微控制器环境设计。其核心功能是执行固定长度&#xff08;10点&#xff09;的滑动平均&#xff08;Moving Average&#xff09;运算&#xff0c;并在每次新采样输入后立即…...

PX4飞控自定义Mavlink消息:实现UART传感器数据在QGC地面站的可视化

1. 为什么需要自定义Mavlink消息 在无人机开发中&#xff0c;我们经常需要将各种传感器数据实时传输到地面站进行监控和分析。PX4飞控虽然内置了丰富的标准Mavlink消息&#xff0c;但当我们接入一些特殊传感器时&#xff0c;标准消息往往无法满足需求。比如你想通过UART串口接入…...

Gumbo-parser内存管理终极指南:7个简单步骤避免常见陷阱

Gumbo-parser内存管理终极指南&#xff1a;7个简单步骤避免常见陷阱 【免费下载链接】gumbo-parser An HTML5 parsing library in pure C99 项目地址: https://gitcode.com/gh_mirrors/gu/gumbo-parser Gumbo-parser是一个纯C99编写的HTML5解析库&#xff0c;高效的内存…...

React Native Interactable跨平台开发终极指南:iOS与Android差异处理技巧

React Native Interactable跨平台开发终极指南&#xff1a;iOS与Android差异处理技巧 【免费下载链接】react-native-interactable Experimental implementation of high performance interactable views in React Native 项目地址: https://gitcode.com/gh_mirrors/re/react…...

ai域名后缀注册对SEO有影响吗

ai域名后缀注册对SEO有影响吗 在当今互联网时代&#xff0c;域名选择对于一个网站的成功至关重要。尤其是对于那些在科技、人工智能&#xff08;AI&#xff09;等前沿领域的企业和个人来说&#xff0c;ai域名后缀注册的问题更是备受关注。本文将从多个角度探讨ai域名后缀注册对…...

wx-dump-4j前端架构解析:React+Ant Design构建现代化管理界面

wx-dump-4j前端架构解析&#xff1a;ReactAnt Design构建现代化管理界面 【免费下载链接】wx-dump-4j 一款基于Java开发的微信数据分析工具。 项目地址: https://gitcode.com/gh_mirrors/wx/wx-dump-4j wx-dump-4j是一款基于Java开发的微信数据分析工具&#xff0c;其前…...

jsTree状态管理插件终极指南:实现用户界面的持久化状态保存

jsTree状态管理插件终极指南&#xff1a;实现用户界面的持久化状态保存 【免费下载链接】jstree jquery tree plugin 项目地址: https://gitcode.com/gh_mirrors/js/jstree jsTree状态管理插件是提升用户体验的关键组件&#xff0c;能够自动保存和恢复树形结构的展开状态…...

深入解析C语言malloc(0)的内存分配机制

1. 深入解析 malloc(0) 的行为机制在 C 语言编程中&#xff0c;内存管理是一个基础但极其重要的话题。malloc 函数作为动态内存分配的核心工具&#xff0c;其行为规范在 C 标准中有明确定义。然而&#xff0c;当我们遇到像 malloc(0) 这样的边界情况时&#xff0c;事情就变得有…...

escodegen浏览器端使用教程:在Web环境中实现代码生成

escodegen浏览器端使用教程&#xff1a;在Web环境中实现代码生成 【免费下载链接】escodegen ECMAScript code generator 项目地址: https://gitcode.com/gh_mirrors/es/escodegen escodegen是一个强大的ECMAScript代码生成器&#xff0c;它能够将抽象语法树(AST)转换回…...

React Native Interactable终极指南:TouchesInside与静态交互对比详解

React Native Interactable终极指南&#xff1a;TouchesInside与静态交互对比详解 【免费下载链接】react-native-interactable Experimental implementation of high performance interactable views in React Native 项目地址: https://gitcode.com/gh_mirrors/re/react-na…...

snabbt.js与Hammer.js集成终极指南:打造流畅触摸手势动画的10个技巧

snabbt.js与Hammer.js集成终极指南&#xff1a;打造流畅触摸手势动画的10个技巧 【免费下载链接】snabbt.js Fast animations with javascript and CSS transforms 项目地址: https://gitcode.com/gh_mirrors/sn/snabbt.js snabbt.js是一个轻量级JavaScript动画库&#…...

开源模型性价比之选:Gemma-3-12b-it在OpenClaw中的实战表现

开源模型性价比之选&#xff1a;Gemma-3-12b-it在OpenClaw中的实战表现 1. 为什么选择Gemma-3-12b-it作为OpenClaw的推理引擎 上个月在优化个人自动化工作流时&#xff0c;我面临一个关键决策&#xff1a;该为OpenClaw选择什么样的大模型作为"大脑"&#xff1f;经过…...

5分钟上手Velocity动态主题动画:让界面动效随用户偏好智能切换

5分钟上手Velocity动态主题动画&#xff1a;让界面动效随用户偏好智能切换 【免费下载链接】velocity Accelerated JavaScript animation. 项目地址: https://gitcode.com/gh_mirrors/ve/velocity Velocity是一款高性能的JavaScript动画库&#xff0c;专注于提供流畅、高…...

Jasny Bootstrap按钮标签组件详解:如何优雅地添加图标标签

Jasny Bootstrap按钮标签组件详解&#xff1a;如何优雅地添加图标标签 【免费下载链接】bootstrap The missing components for your favorite front-end framework. 项目地址: https://gitcode.com/gh_mirrors/boots/bootstrap Jasny Bootstrap作为Bootstrap的扩展组件…...

Vivado报错[Opt 31-430]?别慌,手把手教你从网表里揪出那个‘没爹妈’的FDCE

Vivado报错[Opt 31-430]全流程诊断手册&#xff1a;从网表逆向追踪到代码修复 当Vivado在opt_design阶段抛出[Opt 31-430] Found a FDCE that its data pin is undriven时&#xff0c;多数FPGA开发者的第一反应是检查代码中的寄存器定义。但真实情况往往更复杂——这个报错可能…...

Decision Transformer与行为克隆对比分析:何时选择哪种方法

Decision Transformer与行为克隆对比分析&#xff1a;何时选择哪种方法 【免费下载链接】decision-transformer Official codebase for Decision Transformer: Reinforcement Learning via Sequence Modeling. 项目地址: https://gitcode.com/gh_mirrors/de/decision-transfo…...

ShareList插件开发全攻略:从零开始打造专属网盘工具

ShareList插件开发全攻略&#xff1a;从零开始打造专属网盘工具 【免费下载链接】sharelist 快速分享 GoogleDrive OneDrive 项目地址: https://gitcode.com/gh_mirrors/sh/sharelist ShareList是一款强大的开源网盘工具&#xff0c;支持快速挂载Google Drive、OneDriv…...

跨平台文件同步:OpenClaw+百川2-13B-4bits量化模型智能归档方案

跨平台文件同步&#xff1a;OpenClaw百川2-13B-4bits量化模型智能归档方案 1. 为什么需要智能文件归档 作为一个长期在多台设备间切换工作的开发者&#xff0c;我的文件管理一直处于混乱状态。同一份文档可能同时存在于Mac的Downloads文件夹、Windows桌面的"临时"目…...

高级应用:将Decision Transformer部署到生产环境的完整流程

高级应用&#xff1a;将Decision Transformer部署到生产环境的完整流程 【免费下载链接】decision-transformer Official codebase for Decision Transformer: Reinforcement Learning via Sequence Modeling. 项目地址: https://gitcode.com/gh_mirrors/de/decision-transfo…...

EasyPhoto与ControlNet深度集成:实现精准肖像控制的终极指南

EasyPhoto与ControlNet深度集成&#xff1a;实现精准肖像控制的终极指南 【免费下载链接】sd-webui-EasyPhoto &#x1f4f7; EasyPhoto | Your Smart AI Photo Generator. 项目地址: https://gitcode.com/gh_mirrors/sd/sd-webui-EasyPhoto 在AI肖像生成领域&#xff0…...

别再死记硬背了!用Wireshark抓包实战,5分钟搞懂TCP三次握手和HTTP请求全过程

用Wireshark抓包实战&#xff1a;5分钟可视化TCP三次握手与HTTP请求 刚接触计算机网络时&#xff0c;那些抽象的三次握手、滑动窗口、HTTP报文总让人头晕。直到我第一次用Wireshark看到真实的数据包在屏幕上跳动——原来教科书上的每个概念都能在抓包结果中找到对应的"证…...

5分钟快速上手MUNIT:从零开始构建你的第一个图像翻译模型

5分钟快速上手MUNIT&#xff1a;从零开始构建你的第一个图像翻译模型 【免费下载链接】MUNIT Multimodal Unsupervised Image-to-Image Translation 项目地址: https://gitcode.com/gh_mirrors/mu/MUNIT MUNIT&#xff08;Multimodal Unsupervised Image-to-Image Trans…...

OpenClaw+gemma-3-12b-it:学术论文自动摘要与分类系统

OpenClawgemma-3-12b-it&#xff1a;学术论文自动摘要与分类系统 1. 为什么需要自动化论文处理 作为一名经常需要阅读大量文献的研究者&#xff0c;我深刻体会到手动处理论文的痛点。每周需要从arXiv、PubMed等平台下载数十篇论文&#xff0c;然后逐篇阅读摘要、分类归档。这…...