数据结构之树 --- 二叉树

news/2025/1/17 8:05:35/

目录

定义二叉树的结构体

二叉树的遍历

递归遍历

非递归遍历 

链式二叉树的实现 

二叉树的功能接口

先序遍历创建二叉树

后序遍历销毁二叉树 

 先序遍历查找树中值为x的节点

层序遍历


上篇我们对二叉树的顺序存储堆进行了讲述,本文我们来看链式二叉树。

定义二叉树的结构体

定义链式二叉树同定义链表相同,只是需要注意二叉树有两个指针,类似于双向链表,逻辑上我们将其看作一棵二叉树。下面是定义该树的结构体。

typedef int BTDataType;
typedef struct BinaryTreeNode
{
BTDataType data;
struct BinaryTreeNode* left;
struct BinaryTreeNode* right;
}BTNode;

在创建二叉树之前,我们需要了解前序、中序、后序以及层序遍历。

二叉树的遍历

递归遍历

学习二叉树结构,最简单的方式就是遍历。所谓二叉树遍历(Traversal)是按照某种特定的规则,依次对二叉树中的节点进行相应的操作,并且每个节点只操作一次。访问结点所做的操作依赖于具体的应用问题。 遍历是二叉树上最重要的运算之一,也是二叉树上进行其它运算的基础。

按照规则,二叉树的遍历有:前序/中序/后序的递归结构遍历:
1. 前序遍历(Preorder Traversal 亦称先序遍历)——访问根结点的操作发生在遍历其左右子树之前。
2. 中序遍历(Inorder Traversal)——访问根结点的操作发生在遍历其左右子树之中(间)。
3. 后序遍历(Postorder Traversal)——访问根结点的操作发生在遍历其左右子树之后。

由于被访问的结点必是某子树的根,所以N(Node)、L(Left subtree)和R(Right subtree)又可解释为根、根的左子树和根的右子树。NLRLNRLRN分别又称为先根遍历、中根遍历和后根遍历。

下图展示先序遍历的递归结构:

非递归遍历 

非递归遍历也即层序遍历:

设二叉树的根节点所在层数为1,层序遍历就是从所在二叉树的根节点出发,首先访问第一层的树根节点,然后从左到右访问第2层上的节点,接着是第三层的节点,以此类推,自上而下,自左至右逐层访问树的结点的过程就是层序遍历。

图示: 

链式二叉树的实现 

二叉树的功能接口

数据结构的实现无非是增删查改,二叉树也不例外。

// 通过前序遍历的数组"ABD##E#H##CF##G##"构建二叉树
BTNode* BinaryTreeCreate(BTNode* root, BTDataType* str);
// 二叉树销毁
void BinaryTreeDestory(BTNode** root);
// 二叉树节点个数
int BinaryTreeSize(BTNode* root);
// 二叉树叶子节点个数
int BinaryTreeLeafSize(BTNode* root);
// 二叉树第k层节点个数
int BinaryTreeLevelKSize(BTNode* root, int k);
// 二叉树查找值为x的节点
BTNode* BinaryTreeFind(BTNode* root, BTDataType x);
// 二叉树前序遍历 
void BinaryTreePrevOrder(BTNode* root);
// 二叉树中序遍历
void BinaryTreeInOrder(BTNode* root);
// 二叉树后序遍历
void BinaryTreePostOrder(BTNode* root);层序遍历
//void BinaryTreeLevelOrder(BTNode* root);判断二叉树是否是完全二叉树
//bool BinaryTreeComplete(BTNode* root);

先序遍历创建二叉树

先序遍历创建一个二叉树,我们递归遍历每个元素,然后为其创建节点,但叶子节点没有孩子怎么办?

叶子节点没有孩子,因此当递归到叶子节点的孩子时需要返回NULL;我们需要在需要创建的元素数组中做好标记,比如下面这个代码,当我们遇到 '#' 时返回NULL结束函数,不创建节点。

通过前序遍历的数组"ABD##E#H##CF##G##"构建二叉树
struct BTNode* PreorderCreate(int* a,int* i)
{if (a[*i] == '#'){(*i)++;return NULL;}struct BTNode* root = (struct TreeNode*)malloc(sizeof(struct TreeNode));root->val = a[*i];(*i)++;root->left = PreorderCreate(a, i);root->right = PreorderCreate(a, i);return root;
}

后序遍历销毁二叉树 

销毁一个二叉树,我们设想一下,当销毁了树的根节点,那么我们就找不到他的孩子了。因此根节点必然是最后一个销毁的,所以我们用后序遍历来销毁二叉树。

我们递归遍历最深的节点,依次往上销毁。

// 二叉树销毁(后序遍历)
void BinaryTreeDestory(BTNode** root)
{if (*root == NULL)return;BTNode* cur = *root;BinaryTreeDestory(&(*root)->left);BinaryTreeDestory(&(*root)->right);free(*root);*root = NULL;
}

 先序遍历查找树中值为x的节点

树的查找,这里我们以先序遍历为例。

若该节点数据等于x,则返回该节点。否则递归其孩子,我们分别用ret1与ret2记录递归其左右孩子的返回值,然后判断返回值是否存在。根据函数可知,函数的返回只可能为NULL或者是值为x的节点。

至于我们为什么要使用ret1、ret2,那是因为如果直接用 BinaryTreeFind(root->left, x);来判断的话,我们会再次进入这个节点的递归,造成额外的栈帧负担。

// 二叉树查找值为x的节点
BTNode* BinaryTreeFind(BTNode* root, BTDataType x)
{if (root == NULL)return NULL;if (root->data == x)return root;BTNode* ret1= BinaryTreeFind(root->left, x);if (ret1)return ret1;BTNode* ret2=BinaryTreeFind(root->right, x);if (ret2)return ret2;return NULL;
}

层序遍历

层序遍历为非递归遍历,对于树而言,非递归往往更难。

这里我们借用队列来实现树的层序遍历,关于队列实现层序遍历,我们后面再讲。

// 层序遍历
void BinaryTreeLevelOrder(BTNode* root)
{Queue q;QueueInit(&q);while (QueueEmpty(&q)){if (!root)QueuePush(&q, root);BTNode* tmp = QueueFront(&q);QueuePop(&q);printf("%d ", tmp->data);if (!root->left)QueuePush(&q, root->left);if (root->right)QueuePush(&q, root->right);}	
}


http://www.ppmy.cn/news/1285540.html

相关文章

Halcon颜色通道的处理decompose3/image_to_channels/channels _to _image

Halcon颜色通道的处理 文章目录 Halcon颜色通道的处理一. 图像的通道二. 访问通道1.访问通道2.获取通道的数量 三. 通道分离与合并1. decompose3算子2. image_to_channels 算子3. compose3算子4. channels_to_image算子 四. 处理RGB信息 由于彩色图像通常包含不止一个通道&…

JAVA B/S架构智慧工地源码,PC后台管理端、APP移动端

智慧工地系统充分利用计算机技术、互联网、物联网、云计算、大数据等新一代信息技术,以PC端,移动端,设备端三位一体的管控方式为企业现场工程管理提供了先进的技术手段。让劳务、设备、物料、安全、环境、能源、资料、计划、质量、视频监控等…

【图像分类】【深度学习】【轻量级网络】【Pytorch版本】ShuffleNet_V2模型算法详解

【图像分类】【深度学习】【轻量级网络】【Pytorch版本】ShuffleNet_V2模型算法详解 文章目录 【图像分类】【深度学习】【轻量级网络】【Pytorch版本】ShuffleNet_V2模型算法详解前言ShuffleNet_V2讲解四条实用指导思想G1:相等的通道宽度可以降低存储访问成本G2:大量的分组卷积…

机器学习距离度量方法

1. 机器学习中为什么要度量距离? 机器学习算法中,经常需要 判断两个样本之间是否相似 ,比如KNN,K-means,推荐算法中的协同过滤等等,常用的套路是 将相似的判断转换成距离的计算 ,距离近的样本相…

Python 实现 PDF 到 Word 文档的高效转换(DOC、DOCX)

PDF(Portable Document Format)已成为一种广泛使用的电子文档格式。PDF的主要优势是跨平台,可以在不同设备上呈现一致的外观。然而,当我们需要对文件内容进行编辑或修改,直接编辑PDF文件会非常困难,而且效果…

OfficeWeb365 Indexs 任意文件读取漏洞复现

0x01 产品简介 OfficeWeb365 是专注于 Office 文档在线预览及PDF文档在线预览云服务,包括 Microsoft Word 文档在线预览、Excel 表格在线预览、Powerpoint 演示文档在线预览,WPS 文字处理、WPS 表格、WPS 演示及 Adobe PDF 文档在线预览。 0x02 漏洞概述 OfficeWeb365 /Pi…

学生基本信息管理项目

设计目的 在完成C语言基础内容学习的基础上,结合数组、字符串、结构体与链表等内容,继续综合应用相关理论和编程技术,分析、设计并完成一个可应用的小项目。通过本项目,更为系统地建立知识的关联性,通过对问题的分析和…

如何在Vue.js中使用$emit进行组件通信

Vue.js是一个渐进式JavaScript框架,它以其简洁的数据绑定和组件系统而闻名。在构建具有多个组件层次的Vue应用时,组件间的通信成为一个关键的话题。Vue提供了一种名为$emit的方法,允许子组件向父组件发送消息。本文将详细介绍如何在Vue中使用…