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

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

一、二叉排序树的定义1.1 性质二叉排序树Binary Search TreeBST满足以下性质左子树所有节点的值 根节点的值右子树所有节点的值 根节点的值左右子树本身也是二叉排序树示例text50 / \ 30 70 / \ / \ 20 40 60 801.2 特点中序遍历得到递增有序序列查找效率与树高相关平均时间复杂度 O(log n)最坏 O(n)二、节点定义与基本操作2.1 节点结构ctypedef struct BSTNode { int data; struct BSTNode *left; struct BSTNode *right; } BSTNode, *BSTree;2.2 创建节点cBSTNode* createNode(int value) { BSTNode *node (BSTNode*)malloc(sizeof(BSTNode)); node-data value; node-left NULL; node-right NULL; return node; }三、查找操作3.1 递归实现cBSTNode* search(BSTree root, int key) { if (root NULL || root-data key) { return root; } if (key root-data) { return search(root-left, key); } else { return search(root-right, key); } }3.2 非递归实现cBSTNode* searchIter(BSTree root, int key) { BSTNode *cur root; while (cur ! NULL cur-data ! key) { if (key cur-data) { cur cur-left; } else { cur cur-right; } } return cur; }四、插入操作插入新节点一定在叶子节点位置不会改变树的结构。cBSTree insert(BSTree root, int value) { if (root NULL) { return createNode(value); } if (value root-data) { root-left insert(root-left, value); } else if (value root-data) { root-right insert(root-right, value); } // 值相等时根据需求决定通常不插入 return root; }五、删除操作重点5.1 三种情况删除节点需要分三种情况处理情况描述处理方法情况1删除叶子节点直接删除情况2删除只有左子树或只有右子树的节点用子节点替换情况3删除有左右子树的节点用前驱或后继替换再删除前驱/后继5.2 找前驱和后继前驱左子树中最右边的节点最大值后继右子树中最左边的节点最小值text50 / \ 30 70 / \ / \ 20 40 60 80 节点50的前驱40 节点50的后继605.3 删除逻辑用后继替换c// 找最小值节点 BSTNode* findMin(BSTree root) { while (root-left ! NULL) { root root-left; } return root; } // 删除节点 BSTree deleteNode(BSTree root, int key) { if (root NULL) return NULL; // 查找要删除的节点 if (key root-data) { root-left deleteNode(root-left, key); } else if (key root-data) { root-right deleteNode(root-right, key); } else { // 找到要删除的节点 // 情况1 情况2只有一个孩子或没有孩子 if (root-left NULL) { BSTNode *temp root-right; free(root); return temp; } else if (root-right NULL) { BSTNode *temp root-left; free(root); return temp; } // 情况3有两个孩子用后继替换 BSTNode *minNode findMin(root-right); root-data minNode-data; root-right deleteNode(root-right, minNode-data); } return root; }画图理解删除节点50用后继60替换text删除前 50 / \ 30 70 / \ / \ 20 40 60 80 删除后 60 / \ 30 70 / \ \ 20 40 80六、完整代码演示c#include stdio.h #include stdlib.h typedef struct BSTNode { int data; struct BSTNode *left; struct BSTNode *right; } BSTNode, *BSTree; BSTNode* createNode(int value) { BSTNode *node (BSTNode*)malloc(sizeof(BSTNode)); node-data value; node-left NULL; node-right NULL; return node; } // 查找递归 BSTNode* search(BSTree root, int key) { if (root NULL || root-data key) return root; if (key root-data) return search(root-left, key); return search(root-right, key); } // 查找非递归 BSTNode* searchIter(BSTree root, int key) { BSTNode *cur root; while (cur ! NULL cur-data ! key) { if (key cur-data) cur cur-left; else cur cur-right; } return cur; } // 插入 BSTree insert(BSTree root, int value) { if (root NULL) return createNode(value); if (value root-data) root-left insert(root-left, value); else if (value root-data) root-right insert(root-right, value); return root; } // 找最小值 BSTNode* findMin(BSTree root) { while (root-left ! NULL) root root-left; return root; } // 删除 BSTree deleteNode(BSTree root, int key) { if (root NULL) return NULL; if (key root-data) { root-left deleteNode(root-left, key); } else if (key root-data) { root-right deleteNode(root-right, key); } else { // 情况1 2 if (root-left NULL) { BSTNode *temp root-right; free(root); return temp; } else if (root-right NULL) { BSTNode *temp root-left; free(root); return temp; } // 情况3用后继替换 BSTNode *minNode findMin(root-right); root-data minNode-data; root-right deleteNode(root-right, minNode-data); } return root; } // 中序遍历 void inorder(BSTree root) { if (root NULL) return; inorder(root-left); printf(%d , root-data); inorder(root-right); } // 前序遍历用于看结构 void preorder(BSTree root) { if (root NULL) return; printf(%d , root-data); preorder(root-left); preorder(root-right); } int main() { BSTree root NULL; // 插入节点 int values[] {50, 30, 70, 20, 40, 60, 80}; for (int i 0; i 7; i) { root insert(root, values[i]); } printf(中序遍历有序: ); inorder(root); printf(\n); printf(前序遍历结构: ); preorder(root); printf(\n); // 查找 int key 40; BSTNode *found search(root, key); printf(\n查找 %d: %s\n, key, found ? 找到 : 未找到); // 插入重复值不会插入 root insert(root, 40); printf(插入重复值40后: ); inorder(root); printf(\n); // 删除叶子节点20 root deleteNode(root, 20); printf(\n删除20后: ); inorder(root); printf(\n); // 删除只有一个孩子的节点70 // 先插入一个节点让70只有右孩子 root insert(root, 90); printf(插入90后: ); inorder(root); printf(\n); root deleteNode(root, 70); printf(删除70后: ); inorder(root); printf(\n); // 删除有两个孩子的节点50 root deleteNode(root, 50); printf(删除50后: ); inorder(root); printf(\n); return 0; }运行结果text中序遍历有序: 20 30 40 50 60 70 80 前序遍历结构: 50 30 20 40 70 60 80 查找 40: 找到 插入重复值40后: 20 30 40 50 60 70 80 删除20后: 30 40 50 60 70 80 插入90后: 30 40 50 60 70 80 90 删除70后: 30 40 50 60 80 90 删除50后: 30 40 60 80 90七、删除操作的详细图解以删除根节点50为例text50 ← 要删除 / \ 30 70 / \ / \ 20 40 60 80 步骤1找到后继右子树最小值→ 60 步骤2用60替换50的值 60 / \ 30 70 / \ / \ 20 40 60 80 步骤3删除原来的60节点叶子节点 60 / \ 30 70 / \ \ 20 40 80八、复杂度分析操作平均时间复杂度最坏时间复杂度空间复杂度查找O(log n)O(n)O(1)插入O(log n)O(n)O(1)删除O(log n)O(n)O(1)最坏情况当插入顺序有序时如 1,2,3,4,5BST退化成链表时间复杂度 O(n)。text1 \ 2 \ 3 \ 4九、小结这一篇我们学习了二叉排序树要点说明性质左 根 右中序遍历有序查找递归/非递归O(log n)插入在叶子节点位置插入删除分三种情况重点掌握后继替换缺点可能退化成链表删除的核心逻辑叶子直接删单孩子子节点替换双孩子找后继或前驱替换再删后继下一篇我们讲平衡二叉树AVL树它解决了BST退化成链表的问题。十、思考题为什么BST的中序遍历一定是有序的删除有两个孩子的节点时用前驱替换和用后继替换有什么区别哪种更好如果按升序顺序插入节点BST会变成什么形状时间复杂度是多少尝试实现用前驱替换的删除算法。欢迎在评论区讨论你的答案。

相关文章:

【数据结构与算法】第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;然后逐篇阅读摘要、分类归档。这…...

技术面试终极指南:10个反向面试技巧助你问对公司问题

技术面试终极指南&#xff1a;10个反向面试技巧助你问对公司问题 【免费下载链接】reverse-interview Questions to ask the company during your interview 项目地址: https://gitcode.com/gh_mirrors/re/reverse-interview 在技术面试中&#xff0c;反向面试&#xff…...