5.3二叉树——二叉树链式结构实现

本篇博客梳理二叉树链式结构
明确:二叉树是递归定义
递归的本质:当前问题+子问题,返回条件是最小规模的子问题

一、二叉树的遍历

1.前序、中序与后序遍历

(1)前序:根->左子树->右子树(每个子树也满足这个遍历顺序,下同)
(2)中序:左子树->根->右子树
(3)后序:左子树->右子树->根
分析前序遍历
前序遍历过程
递归展开图如下,红色箭头表示递推,绿色箭头表示回归
前序遍历递归展开图

// 二叉树前序遍历:根-左子树-右子树
void BinaryTreePrevOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}printf("%d ", root->data);BinaryTreePrevOrder(root->leftChild);BinaryTreePrevOrder(root->rightChild);
}

前序遍历结果:
前序遍历结果
1的左右子树是两个红框,红框内的树仍旧满足前序遍历

// 二叉树中序遍历:左子树-根-右子树
void BinaryTreeInOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}BinaryTreeInOrder(root->leftChild);printf("%d ", root->data);BinaryTreeInOrder(root->rightChild);
}

中序遍历结果:
中序遍历结果

// 二叉树后序遍历:左子树-右子树-根
void BinaryTreePostOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}BinaryTreePostOrder(root->leftChild);BinaryTreePostOrder(root->rightChild);printf("%d ", root->data);
}

后序遍历结果:
后序遍历结果

2.层序遍历(广度优先遍历【BFS】)

逐层访问二叉树的结点
层序遍历
① 算法思想:用队列辅助,上一层带下一层
② 具体操作:队列结点的data存指向二叉树结点的指针,一个结点出列,该节点孩子马上入列(空结点不入列)
③ 画图分析:
层序遍历画图分析
代码实现如下,队列相关的函数可在4.1中找到

// 层序遍历
//用队列辅助,每个节点当中存指向二叉树对应结点的指针
void BinaryTreeLevelOrder(BTNode* root)
{Queue queue;QueueInit(&queue);//注意:队列中链表节点的data要改成BTNode*的指针//根节点先入列QueuePush(&queue, root);while (!QueueEmpty(&queue)){//一个结点出列,带着其孩子入列,空结点不入列BTNode* front = QueueFront(&queue);if (front->leftChild != NULL)//左孩子不为空则入列{QueuePush(&queue, front->leftChild);}if (front->rightChild != NULL)//右孩子不为空则入列{QueuePush(&queue, front->rightChild);}printf("%d ", QueueFront(&queue)->data);QueuePop(&queue);//结点出列}QueueDestroy(&queue);
}

二、结点个数与高度等

二叉树链式结构

typedef int BTDataType;
typedef struct BinaryTreeNode
{BTDataType data;struct BinaryTreeNode* leftChild;struct BinaryTreeNode* rightChild;
}BTNode;

1.二叉树结点个数:int BinaryTreeSize(BTNode* root);

节点个数返回值如下:

  • 空:return 0
  • 不为空:return 左子树结点数+右子树结点数+1
// 二叉树节点个数
int BinaryTreeSize(BTNode* root)
{if (root == NULL)return 0;if (root->leftChild == NULL && root->rightChild == NULL)return 1;elsereturn BinaryTreeSize(root->leftChild) +BinaryTreeSize(root->rightChild) + 1;
}

2.二叉树叶子结点个数:int BinaryTreeLeafSize(BTNode* root);

叶子结点个数返回值如下

  • 空:return 0
  • 叶子:return 1
  • 非叶子:return 左子树叶子数+右子树叶子数
//二叉树叶子节点个数
int BinaryTreeLeafSize(BTNode* root)
{if (root == NULL)return 0;if (root->leftChild == NULL && root->rightChild == NULL)return 1;return BinaryTreeLeafSize(root->leftChild)+ BinaryTreeLeafSize(root->rightChild);
}

3.二叉树第k层结点个数int BinaryTreeLevelKSize(BTNode* root, int k);

  • 空:return 0
  • 非空且k==1:return 1
  • 非空且k>1:研究左子树第k-1层+右子树第k-1层
// 二叉树第k层节点个数
int BinaryTreeLevelKSize(BTNode* root, int k)
{if (root == NULL)return 0;if (root && k == 1)return 1;return BinaryTreeLevelKSize(root->leftChild, k - 1) +BinaryTreeLevelKSize(root->rightChild, k - 1);
}

4.判断二叉树是否为完全二叉树:int BinaryTreeComplete(BTNode* root);

① 算法思想:层序遍历
② 具体操作:层序遍历,把空也带入队列,第一个空出列之后就开始检查,如果队列中还有非空元素,就不是完全二叉树

//判断二叉树是否是完全二叉树
int BinaryTreeComplete(BTNode* root)
{if (root == NULL)return 0;//用层序遍历,把所有数据(包括NULL)也带入队列//当第一个空出列之后,开始判断,如果队列中还有非空就不是完全二叉树Queue queue;QueueInit(&queue);QueuePush(&queue, root);//根先入队列while (!QueueEmpty(&queue)){BTNode* front = QueueFront(&queue);if (front != NULL){QueuePop(&queue);QueuePush(&queue, front->leftChild);QueuePush(&queue, front->rightChild);}//遇到第一个NULL在队头就开始检查if (front == NULL){break;}}//注意:NULL指针在队列当中,队列不是空while (!QueueEmpty(&queue)){BTNode* front = QueueFront(&queue);if (front == NULL){QueuePop(&queue);}if (front != NULL)//发现队列当中还有不为NULL的元素,就不是完全二叉树return 0;}return 1;
}

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

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

相关文章

书生大模型实战营(1)——InterStudio基础知识+Vscode SSH连接远程服务器+Linux基础指令

参加书生.浦江大模型实战训练营,学习大模型知识和微调技术,所有课程免费,通过闯关的形式学习,也比较有趣。一起来了解LLM的世界。邀请链接 产品简介 InternStudio 是大模型时代下的云端算力平台。基于 InternLM 组织下的诸多算法…

经典文献阅读之--ParkingE2E(基于摄像头的端到端停车网络:从图像到规划)

0. 简介 自动泊车是智能驾驶领域的一项关键任务。传统泊车算法通常采用基于规则的方案来实现。然而,由于算法设计的复杂性,这些方法在复杂的泊车场景中效果欠佳。相比之下,基于神经网络的方法往往比基于规则的方法更加直观且功能多样。通过收…

ORACLE 统计信息的备份与恢复

备份 --需要先创建统计信息基础表 exec dbms_stats.create_stat_table(USER1,STAT_TIMESTAMP); --导出某个用户的所有统计信息 exec dbms_stats.export_schema_stats(USER1,STAT_TIMESTAMP);--测试(插入100条,更新统计信息,略) select num_rows,last_ana…

基于STM32开发的简易自动驾驶系统

目录 引言环境准备工作 硬件准备软件安装与配置系统设计 系统架构硬件连接代码实现 系统初始化传感器数据采集与处理电机控制与转向OLED显示与状态提示Wi-Fi通信与远程监控应用场景 简易自动驾驶演示智能车模型开发与学习常见问题及解决方案 常见问题解决方案结论 1. 引言 随…

蜂鸣器奏乐

一、粗略了解简谱 拍号:如图,“2”表示一个小节有2拍,“4”表示4分音符为一拍 终止线表示歌曲结束 注意:以下音符都按以四分音符为一拍计算拍数 四分音符: 唱一拍 二分音符: 某一个音右边有一个小横线&…

中国招标投标平台JS逆向:DES加密与Python纯算还原

中国招标投标平台JS逆向:DES加密与Python纯算还原 目录 🔐 JS DES解密🧮 Python版本的纯算实现 🔐 JS DES解密 在中国招标投标公共服务平台的分析过程中,发现了数据加密采用了DES算法。DES(数据加密标准&…

github源码指引:C++嵌入式WEB服务器

初级代码游戏的专栏介绍与文章目录-CSDN博客 我的github:codetoys,所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。 这些代码大部分以Linux为目标但部分代码是纯C的,可以在任何平台上使用。 相关专题: C嵌入式…

Broker服务器模块

一.Broker模块介绍 二.Broker模块具体实现 1. 类的成员变量与构造函数 成员变量 事件循环和TCP服务器: muduo::net::EventLoop _baseloop;muduo::net::TcpServer _server; 这些是muduo库提供的核心组件,负责处理网络事件和管理TCP连接。 消息分发和编码: muduo::…

Spring Security 认证源码超详细分析

Spring Security 认证源码超详细分析 认证(Authentication)是系统确认用户信息的重要途径,用户通过认证之后,系统才能明确用户的身份,进而才可以为该用户分配一定的权限,这个过程也叫授权(Auth…

项目实战--Sa-Token详细方案笔记

Sa-Token权限系统设计 一、前言二、认证授权的概念三、Sa-Token简介3.1 Sa-Token使用方式3.2 踢人下线3.3 全局异常处理3.4 二级认证3.5 同端互斥登录3.6 Http Basic/Digest 认证3.6.1 HttpBasic认证3.6.2 Http Digest 认证 四、Sa-Token授权(鉴权)4.1 权…

详说 类和对象

类怎么定义 类是什么呢?类就是我们上篇文说的命名空间,单独创建一个域,自己有自己的生命空间,那么类怎么定义呢?C规定,假设 stack就是他的类名,那么前面要加个class,换行之后就是他…

汽车乘客热舒适度大挑战,如何利用仿真技术提高汽车环境舒适度

舒适性在人们选择汽车的决定性方面占比越来越重,而汽车乘员舱环境的舒适性是指为乘员提供舒适愉快便利的乘坐环境与条件,包括良好的平顺性、车内的低噪声、适宜的空气环境以及良好的驾驶操作性能。 舒适性 经济性 安全性、动力性 典型的乘员舱热舒适性模…

laravel的队列的使用

laravel队列 laravel的特性:laravel队列可以基于不同的后台存储服务提供统一的api,后台存储服务包括 Redis MySQL等。队列实现了业务解耦,异步处理,错误重试的功能。比如调用第三方api,无法保证api的可靠性&#xff0…

Transformer 与传统模型Informer

Transformer 与传统模型:Informer 如何改变时间序列预测的规则 Transformers 是那些聪明的注意力构建者,它们在机器学习的各个领域掀起了波澜。但在时间序列预测领域,它们才真正大显身手。你可能会问,为什么?想象一下,有一个水晶球,它不仅能看到未来,还能理解导致未来的…

TCP协议 配合 Wireshark 分析数据

在TCP连接中,无论是客户端还是服务端,都有可能成为发送端或接收端,这是因为TCP是一个全双工协议,允许数据在同一连接中双向流动 客户端(Client):通常是指主动发起连接请求的一方。例如&#xf…

宠物空气净化器有用吗?为什么养宠家庭要买宠物空气净化器?

身为一个鼻炎患者,却喜欢猫咪,所以毅然决然的养了两只宠物,而且还是长毛猫,不要问为什么鼻炎还买两只猫咪,因为怕一只猫咪孤单,所以养了两只。对于很多人来说,猫咪就像焦虑不安时的精神搭子&…

如何让私域服务赢得用户的心?

私域流量的概念在当今的商业环境中已经变得极为重要,许多品牌和企业都投入大量资源尝试通过各种策略吸引并保留用户。然而,单纯的流量积累并不足以确保商业成功。当面对用户的沉默、缺乏活跃度以及无法变现的困境时,我们必须重新审视私域流量…

语音控制开关的语音识别ic芯片方案

语音控制开关是一种基于语音识别技术的设备,它通过内置的语音识别芯片,将用户的语音指令转化为电信号,从而实现对设备的控制。例如在智能家居设备上的应用,通常需要连接到家庭的Wi-Fi网络上,以便与智能手机或智能音箱等…

Java之初始泛型

1 包装类 在Java中,由于基本类型不是继承自Object,为了在泛型代码中可以支持基本类型,Java给每个基本类型都对应了一个包装类型。 1.1 基本数据类型和对应的包装类 基本数据类型包装类byteByteshortShortintIntegerlongLongfloatFloatdoub…

FaceFormer嘴形同步论文复现

一、项目地址 https://github.com/EvelynFan/FaceFormer 二、复现过程 1、项目环境 系统:Ubuntu 18.04.1 python版本:Python 3.7 使用conda创建一个虚拟环境,安装requirements.txt中所需要的库 2、安装ffmpeg 教程网址:http…