西风的数据结构教程(3)——二叉树

2024-02-23 13:18

本文主要是介绍西风的数据结构教程(3)——二叉树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今天我们终于要说到计算机界的经典数据结构——树,树是非常良好的一种组织数据的形式,在现实生活中也经常用到。

树的结构,大概就是这个样子:

这里写图片描述

一群树就会组成一个森林

不过,这样并不好管理,所以我们一般会还是使用一个根节点来管理它们。

树结构最大的特点就是,一个节点的情况分析只和父节点和子节点相关,这样管理数据时往往不需要考虑很多。

例如文件系统中,每个文件夹可以视为一个节点,而每个文件夹只显示自己的子节点,结构就非常清晰了。

而二叉树是树结构中的非常常用的一种,往往被用作排序树使用,可以自动排序和判重,可以作为索引使用。

下面我们就来实际写一个二叉排序树,然后增加其平衡性,将其改写为一棵红黑树,不过红黑树的部分十分复杂,可以循序渐进的来学习和理解,不必一下就学习完。

二叉排序树

二叉排序树是简单的排序数据结构,可以支持动态的插入删除数据,对数据快速查找,复杂度只有log(n)

它的实现方法是这样的:

这里写图片描述

原理很简单,对于每一棵子树,左孩子都比自己小,右孩子都比自己大,那么就利用二分的思想,不断寻找这个数所在的位置。

插入时,则需要先考虑好要插入的位置:

这里写图片描述

我们发现,插入操作和查找操作是几乎一样的,只要找到对应的位置,插入很好办。

删除时,确实要复杂一点,因为我们直接将节点从树上拿走,如果叶子节点还好办,但我们考虑非叶子节点的删除,就较为复杂了。那么,如何才能正确的删除一个节点呢?我们来观察一下:

这里写图片描述

通过这张图,我们能看出来,每一个元素的上一个相邻元素是左子树的最右边孩子,下一个元素是右子树的最左孩子,这两个相邻元素都是叶子节点,而且如果左子树没有右孩子节点,那么就用最大的根节点替换。同理,用右子树删除也可以。

这里写图片描述

那好,我们既然知道了树的基本操作,就来简要的模拟一下,二叉排序树是如何运作的。

二叉排序树的实现

让我们来定义一棵树吧

typedef int TreeElementType;typedef struct _bst_node
{TreeElementType data;struct _bst_node *left, *right;
} bst_node;typedef struct _tree
{bst_node* root;
} bstree;

还是类似的方法,bstree存抽象的树对象,bst_node则是树的每个节点,其中存放的元素单独定义出来

为其编写构造函数:


bst_node* TreeNodeCreate() {bst_node* n = (bst_node*) malloc(sizeof(bst_node));n->left = n->right = NULL;
}bstree* BSTreeCreate() {return (bstree*)malloc(sizeof(bstree));
}

下面我们要开始实现这个二叉树了,二叉树的插入十分简单,只需要递归的判断,当前节点是比中值大还是小,如果小就往左走,大就往右走,相同时就直接返回无法插入即可。
但这样实现考虑到函数的调用效率问题,我们采用while循环来实现,只需要用指针模拟两个节点即可:

void TreeInsert(bstree* t, TreeElementType data) {if (t->root == NULL) { t->root = TreeNodeCreate();t->root->data = data; return;}bst_node* p = t->root;bst_node* q = NULL;while(p != NULL) {if (data < p->data) { q = p; p = p->left; continue; }if (data > p->data) { q = p; p = p->right; continue; }if (data ==p->data) return;}bst_node* node = TreeNodeCreate();node->data = data;if (data < q->data)q->left = node;elseq->right = node;
}

而查找方法也将是使用循环实现的:

bst_node* TreeSearch(bstree* t, TreeElementType data) {if (t->root == NULL) return NULL;bst_node* p = t->root;while(p != NULL) {if (data ==p->data) return p;if (data < p->data) { p = p->left; continue; }if (data > p->data) { p = p->right; continue; }}return NULL;
}

二叉树的删除是最复杂的部分,核心思路就是找一个可以和当前节点互换的元素,将那个元素替换上去,当前节点拿下来:

void TreeDelete(bstree* t, TreeElementType data) {if (t->root == NULL) return;bst_node* p = t->root;bst_node* q = NULL;while(p != NULL) {if (data ==p->data) break;if (data < p->data) { q = p; p = p->left; continue; }if (data > p->data) { q = p; p = p->right; continue; }}if (p == NULL) return;if (p->left == NULL) { // 若p没有左孩子,直接用它的右孩子取代q->right = p->right;Free(p);} else { // 否则先找到做子树中最大的元素ptemp, 用它交换p, 然后ptemp由ptemp的左孩子代替(因为它一定没有右孩子)bst_node* ptemp = p->left;bst_node* qtemp = NULL;while(ptemp->right != NULL) {qtemp = ptemp;ptemp = ptemp->right;}q->right = ptemp;qtemp->right = ptemp->left;ptemp->left = p->left;ptemp->right = p->right;Free(p);}
}

本系列教程由 西风逍遥游 原创,转载请注明出处:http://blog.csdn.net/xfxyy_sxfancy

这篇关于西风的数据结构教程(3)——二叉树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/738764

相关文章

C#集成DeepSeek模型实现AI私有化的流程步骤(本地部署与API调用教程)

《C#集成DeepSeek模型实现AI私有化的流程步骤(本地部署与API调用教程)》本文主要介绍了C#集成DeepSeek模型实现AI私有化的方法,包括搭建基础环境,如安装Ollama和下载DeepS... 目录前言搭建基础环境1、安装 Ollama2、下载 DeepSeek R1 模型客户端 ChatBo

JAVA集成本地部署的DeepSeek的图文教程

《JAVA集成本地部署的DeepSeek的图文教程》本文主要介绍了JAVA集成本地部署的DeepSeek的图文教程,包含配置环境变量及下载DeepSeek-R1模型并启动,具有一定的参考价值,感兴趣的... 目录一、下载部署DeepSeek1.下载ollama2.下载DeepSeek-R1模型并启动 二、J

Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)

《Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)》文章介绍了如何使用dhtmlx-gantt组件来实现公司的甘特图需求,并提供了一个简单的Vue组件示例,文章还分享了一... 目录一、首先 npm 安装插件二、创建一个vue组件三、业务页面内 引用自定义组件:四、dhtmlx

MySQL zip安装包配置教程

《MySQLzip安装包配置教程》这篇文章详细介绍了如何使用zip安装包在Windows11上安装MySQL8.0,包括下载、解压、配置环境变量、初始化数据库、安装服务以及更改密码等步骤,感兴趣的朋... 目录mysql zip安装包配置教程1、下载zip安装包:2、安装2.1 解压zip包到安装目录2.2

Java使用Tesseract-OCR实战教程

《Java使用Tesseract-OCR实战教程》本文介绍了如何在Java中使用Tesseract-OCR进行文本提取,包括Tesseract-OCR的安装、中文训练库的配置、依赖库的引入以及具体的代... 目录Java使用Tesseract-OCRTesseract-OCR安装配置中文训练库引入依赖代码实

Ubuntu固定虚拟机ip地址的方法教程

《Ubuntu固定虚拟机ip地址的方法教程》本文详细介绍了如何在Ubuntu虚拟机中固定IP地址,包括检查和编辑`/etc/apt/sources.list`文件、更新网络配置文件以及使用Networ... 1、由于虚拟机网络是桥接,所以ip地址会不停地变化,接下来我们就讲述ip如何固定 2、如果apt安

PyCharm 接入 DeepSeek最新完整教程

《PyCharm接入DeepSeek最新完整教程》文章介绍了DeepSeek-V3模型的性能提升以及如何在PyCharm中接入和使用DeepSeek进行代码开发,本文通过图文并茂的形式给大家介绍的... 目录DeepSeek-V3效果演示创建API Key在PyCharm中下载Continue插件配置Con

Deepseek R1模型本地化部署+API接口调用详细教程(释放AI生产力)

《DeepseekR1模型本地化部署+API接口调用详细教程(释放AI生产力)》本文介绍了本地部署DeepSeekR1模型和通过API调用将其集成到VSCode中的过程,作者详细步骤展示了如何下载和... 目录前言一、deepseek R1模型与chatGPT o1系列模型对比二、本地部署步骤1.安装oll

在不同系统间迁移Python程序的方法与教程

《在不同系统间迁移Python程序的方法与教程》本文介绍了几种将Windows上编写的Python程序迁移到Linux服务器上的方法,包括使用虚拟环境和依赖冻结、容器化技术(如Docker)、使用An... 目录使用虚拟环境和依赖冻结1. 创建虚拟环境2. 冻结依赖使用容器化技术(如 docker)1. 创

Go语言中三种容器类型的数据结构详解

《Go语言中三种容器类型的数据结构详解》在Go语言中,有三种主要的容器类型用于存储和操作集合数据:本文主要介绍三者的使用与区别,感兴趣的小伙伴可以跟随小编一起学习一下... 目录基本概念1. 数组(Array)2. 切片(Slice)3. 映射(Map)对比总结注意事项基本概念在 Go 语言中,有三种主要