数据结构之二叉树的精讲

𝙉𝙞𝙘𝙚!!👏🏻‧✧̣̥̇‧✦👏🏻‧✧̣̥̇‧✦ 👏🏻‧✧̣̥̇:Solitary_walk

      ⸝⋆   ━━━┓
     - 个性标签 - :来于“云”的“羽球人”。 Talk is cheap. Show me the code
┗━━━━━━━  ➴ ⷯ

本人座右铭 :   欲达高峰,必忍其痛;欲戴王冠,必承其重。

👑💎💎👑💎💎👑 
💎💎💎自💎💎💎
💎💎💎信💎💎💎
👑💎💎 💎💎👑    希望在看完我的此篇博客后可以对你有帮助哟

👑👑💎💎💎👑👑   此外,希望各位大佬们在看完后,可以互相支持,蟹蟹!
👑👑👑💎👑👑👑


对二叉树的基本概念性的理解,若有不明白的友友们,可以看一下前期写的关于堆与二叉树的精讲

链接在此:

有了大家对二叉树的基本理解接下来,对以下知识的理解应该是易如反掌!

1.二叉树的链式结构的实现

 对于任何一个二叉树的节点来说:都是由值域,左孩子,右孩子构成,只不过对于某些节点而言左右孩子为空

1.1定义一个二叉树的结构体
typedef int BT_DataType;
typedef struct BinaryTreeNode
{struct BinaryTreeNode* left; //左孩子struct BinaryTreeNode* right;//右孩子BT_DataType val;  //值域}BT;
1.2二叉树的链式结构

 为了方便对二叉树的理解,我暂时手动创建节点,进行连接

BT* BT_Create()
{BT* n1 = BuyNode( 1);BT* n2 = BuyNode( 2);BT* n3 = BuyNode( 3);BT* n4 = BuyNode( 4);BT* n5 = BuyNode(5);BT* n6 = BuyNode( 6);BT* n7 = BuyNode( 7);BT* n8= BuyNode( 8);BT* n9 = BuyNode( 9);BT* n10 = BuyNode( 10);n1->left = n2;n1->right = n3;n2->right = n4;n3->left = n5;n3->right = n6;n2->left = n7;n4->left = n8;return n1;
}
2.二叉树的遍历
2.1前序遍历(先序遍历)

先访问根节点,其次访问左子树,左后访问右子树,注意这是一个递归的定义。

见图如下:

 代码:
void Pre_order(BT* root)
{if (root == NULL){printf("%c ", 'n');//返回n表示为空return;}printf("%d ", root->val);Pre_order(root->left);Pre_order(root->right);
}
递归展开图的解释:

2.2中序遍历

先访问左子树,在访问根节点最后访问右子树,依然是一个递归的定义

分析如下:
 代码:
void In_Order(BT* root)
{if (root == NULL){printf("%c ", 'n');//返回n表示为空return;}In_Order(root->left);printf("%d ", root->val);In_Order(root->right);
}
2.3后续遍历

先访问左子树,再访问右子树,最后访问根节点,依然是递归定义

分析见下:

代码:
void Post_Order(BT* root)
{if (root == NULL){printf("%c ", 'n');//返回n表示为空return;}Post_Order(root->left);Post_Order(root->right);printf("%d ", root->val);}
3.与二叉树相关节点的求解
3.1求二叉树所有节点个数

可能有些老铁们会说:直接进行计数就可以了:只有是当前节点不为空就让计数器(size)加一,采用前序遍历的方法。没错也可以这样但是这样有些坑需要注意一下否则一不小心就掉进坑里了。

 

 这时候可能有老铁会说,直接定义一个全局变量不就OK了,注意当我们连续调用BT_Size()这个函数进行求个数的话会有问题滴!

 因为szie作为一个全局变量,第一调用确实为8没有错,但是当我们后续在进行调用的时候就需要把size 手动进行置零,(关于这个问题详解,感兴趣的友友们,可以看前期我写的博客:链接在此,自取,蟹蟹支持!)要不然只会在当前size = 8 的基础上进行相加,这也就是为什么最后会出现16,24的这样结果了,也就不以为奇了。

代码:
int size = 0;
int BT_Size(BT* root)
{//int size = 0;if (root == NULL)return size;size++;BT_Size(root->left);//对左子树进行个数的遍历BT_Size(root->right);//对右子树进行个数的遍历return size;
}

 运行结果:

3.2求二叉树叶节点个数

既然是让咱求叶节点个数,咱就需要知道什么是叶节点:度为0的节点(没有左孩子,也没有右孩子的节点)

通过前面对二叉树的遍历咱们应该渐渐对递归要有些体悟了,当一个问题很大的时候,可以化大为小,化繁为简。这样岂不是很爽!

 分析见下:
 代码:
int BT_Leaf_Size(BT* root)
{if (root == NULL)return 0;if (root->left == NULL && root->right == NULL) // 判断是否为叶节点return 1;return  BT_Leaf_Size(root->left) + BT_Leaf_Size(root->right);
}
3.3求二叉树第H层节点个数

假设根节点所在层次就是第一层,依次下推,第H层的每个节点必然是第(H-1)层节点的左右孩子,这不就解决问题了嘛:求二叉树第H层节点个数转化为求二叉树第H-1层每个节点的左右孩子之和不就OK了。

 分析见下:

 代码:
int BT_LevelH_Size(BT* root, int h)
{if (root == NULL)return 0;if (h == 1)return 1;
return BT_LevelH_Size(root->left, h - 1)+ BT_LevelH_Size(root->right, h - 1) ;
}
4.二叉树高度的求解

对于一个二叉树而言,树的高度是左子树与右子树相比高度最大的一个再加1

还是如此,借助递归思想

子问题:左子树与右子树相比高度最大的一个再加1

结束条件:判断当前节点是否为空,其次当前节点是否为叶节点

 分析见下:

代码:
int BT_Height(BT* root)// 求树高度
{if (root == NULL)return 0;if (root->left == NULL && root->right == NULL)return 1;int left_h = BT_Height(root->left);int right_h = BT_Height(root->right);return left_h > right_h ? left_h + 1 : right_h + 1;
}
5.二叉树的销毁

注意这是链式二叉树,不能直接进行删除,更为直接的是采用后续遍历来进行销毁

代码:

void BT_Destroy(BT* root)
{if (root == NULL)return;BT_Destroy(root->left);BT_Destroy(root->right);free(root);
}
6.二叉树的层次遍历

 对于二叉树的层次遍历很显然就是一层一层进行遍历,在这里借助队的性质先进先出,来实现对二叉树的层次遍历

当队头元素出队的时候同时让队头元素的左右孩子节点也进队

 这里需要借助咱们之前写的队列的相关代码才可以玩哟!

 代码:
void Level_order(BT* root)
{Queque q;QuqueInit(&q);QuquePush(&q, root);while (!QuequeEmpty(&q)){BT* front = QuequeFront(&q);//取队头元素if (front)printf("%d ", front->val);QuquePop(&q);//删除队头元素//队头元素的左右孩子进队if (front)  {QuquePush(&q, front->left);QuquePush(&q, front->right);}}QuqueDestroy(&q);
}
7.借助二叉树前序遍历的结果实现对二叉树的构建

 分析: 还是分治的思想,层层递归,直到遇到空返回,把每一个节点看作一个新的根节点,只要当前节点不为空,就继续先序遍历

首先这是一个IO型的,所以完全需要自己手撕代码,

先把当前字符串的内容进行接收

其次利用递归:先序建立二叉树

子问题:先序建立    结束条件:遇到空,直接返回

最后利用递归写一个中序遍历

 辅助:需要我们每一个数据开辟节点

 定义一个二叉树的结构体:

 递归建立二叉树:

注意这里必须是 (*pi)++,不能是 *pi++。因为是递归调用每一次都需要进行栈帧建立,这样就不能做保证下标正确性,问题本质同二叉树求节点个数中的size

 完整代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>typedef char BT_DataType; // 存储char类型数据
typedef struct BinaryTreeNode
{struct BinaryTreeNode* left;struct BinaryTreeNode* right;BT_DataType val;}BT;
BT* BuyNode( BT_DataType x)
{BT*  n1 = (BT*)malloc(sizeof(BT));if (n1 == NULL)return NULL;n1->val = x;n1->left = n1->right = NULL;return n1;
}
BT* CreateTree(char*pa,int* pi){if(pa[*pi] == '#'){(*pi)++; // err *(pi)++return NULL;}BT*root = BuyNode(pa[*pi]);(*pi)++; // err *(pi)++root->left = CreateTree(pa,pi );root->right =CreateTree(pa,pi );return root;}void  In_Order(BT*root){if(root == NULL)return ;In_Order(root->left);printf("%c ",root->val);In_Order(root->right);}int main() 
{char a[100];scanf("%s",a);int i = 0;// 下标访问数组BT* root =  CreateTree(a,&i);In_Order(root);
}
8.判断一棵树是否为完全二叉树

 对于这个,可以借助层次遍历的思想来玩。只要是在出队的时候连续全部为空即为完全二叉树;否则就不是完全二叉树

 

代码:

bool BT_Complete(BT* root)
{Queque q;QuqueInit(&q);QuquePush(&q, root);while (!QuequeEmpty(&q)){BT* front = QuequeFront(&q);//取队头元素QuquePop(&q);//删除队头元素if (front == NULL)break;QuquePush(&q, front->left);QuquePush(&q, front->right);}//来到这只能有2种情况:队列为空  front == NULLwhile (!QuequeEmpty(&q)){//只能是front为空BT* front1 = QuequeFront(&q);//取队头元素if(front1)return false;  //非空 说明不是二叉树QuquePop(&q);}QuqueDestroy(&q);return true;
}
9:二叉树的查找

查找某个节点的值是否存在,若存在则返回该节点的地址,否则返回NULL

可以用先序来遍历

 错误示例:当我想查找3这个节点时候

 相信有不少老铁们一开始就会这样写吧,但是明明3这个节点存在为什么会没有找到呢?

其实通过我们调试发现这样写的逻辑是有Bug的,及时当要查找的节点存在时,也会造成把已找到的节点覆盖掉,其实这个查找逻辑的代码重在对return 返回的考察

正确代码:
BT* BT_Find(BT* root, BT_DataType x) // 3
{if (root == NULL)return  NULL;if (root->val == x)return root; //先去左子树查找BT* left = BT_Find(root->left, x);if (left)return left;//说明左子树不存在,去右子树查找BT* right = BT_Find(root->right, x);if (right)return right;//来到这说明左右子树都不存在return NULL;
}

 运行结果:

10.二叉树相关OJ的练习
10.1 单值二叉树

 解题分析:

其实一个遍历就直接搞定了。拿双亲节点的值与孩子节点对应的值进行比较,若是不相等直接 return false

是不是看着比较简单,但是写起来是有坑滴

 

 运行结果:

其实走读一遍代码大概就找到问题所在了:return 语句返回是返回到调用当前函数的上一个函数里,并不是直接返回到最外层 

这个问题的本质同二叉树查找指定数据是一样的

OJ代码:
bool isUnivalTree(struct TreeNode* root) 
{if(root == NULL)return true;if(root->left){if(root->val != root->left->val)return false;}if(root->right){if(root->val != root->right->val)return false;}//递归左子树bool ret1 = isUnivalTree(root->left);//递归右子树bool ret2=  isUnivalTree(root->right);return ret1 && ret2;
}
10.2 判断2个二叉树是否相同

 解题分析:

采用遍历的方式,根节点与根节点进行对比,左孩子与左孩子对比,右孩子与右孩子对比,注意是对比val而不是对比节点所对应的地址

OJ代码:
bool isSameTree(struct TreeNode* p, struct TreeNode* q){if(p == NULL &&q == NULL)return true;if(p == NULL || q == NULL)return false;//来到这说明都不为空if(p->val  != q->val)return false;return  isSameTree(p->left,q->left) && isSameTree(p->right,q->right);//注意这里必须是 逻辑且的关系,不能是逻辑或
}

 OK,以上就是我今日要为大家分享的,希望各位都有得!对于二叉树这部分是数据结构初阶比较难的一部分了,初学的时候是不好理解,事事都有个过渡期,当然自己有空的时候也不要忘了敲敲代码!

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

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

相关文章

网络编程学习

思维导图 代码练习 TCP实现通信 服务器端代码 #include <myhead.h> #define SER_IP "192.168.152.135" #define SER_PORT 8910 int main(int argc, const char *argv[]) {//&#xff11;创建用于监听的套接字int sfd -1;sfd socket(AF_INET,SOCK_STREAM,0)…

【刷题】Leetcode 1609.奇偶树

Leetcode 1609.奇偶树 题目描述广度优先搜索&#xff08;BFS&#xff09;深度优先算法&#xff08;DFS&#xff09; 思路一&#xff08;BFS&#xff09;思路二&#xff08;DFS&#xff09;Thanks♪(&#xff65;ω&#xff65;)&#xff89;谢谢阅读&#xff01;&#xff01;&a…

网络爬虫的危害,如何有效的防止非法利用

近年来&#xff0c;不法分子利用“爬虫”软件收集公民隐私数据案件屡见不鲜。2023年8月23日&#xff0c;北京市高级人民法院召开北京法院侵犯公民个人信息犯罪案件审判情况新闻通报会&#xff0c;通报侵犯公民个人隐私信息案件审判情况&#xff0c;并发布典型案例。在这些典型案…

FMM 笔记:st-matching(colab上执行)【官方案例解读】

在colab上运行&#xff0c;所以如何在colab上安装fmm&#xff0c;可见FMM 笔记&#xff1a;在colab上执行FMM-CSDN博客 st-matching见论文笔记&#xff1a;Map-Matching for low-sampling-rate GPS trajectories&#xff08;ST-matching&#xff09;-CSDN博客 0 导入库 from…

【vuex之五大核心概念】

vuex:五大核心概念 一、state状态1.state的含义2.如何访问以及使用仓库的数据&#xff08;1&#xff09;通过store直接访问获取store对象 &#xff08;2&#xff09;通过辅助函数MapState 二、mutations1.作用2.严格模式3.操作流程定义 mutations 对象&#xff0c;对象中存放修…

Parquet 文件生成和读取

文章目录 一、什么是 Parquet二、实现 Java 读写 Parquet 的流程方式一&#xff1a;遇到的坑&#xff1a;坑1&#xff1a;ClassNotFoundException: com.fasterxml.jackson.annotation.JsonMerge坑2&#xff1a;No FileSystem for scheme "file"坑3&#xff1a;与 spa…

第四十三天| 1049. 最后一块石头的重量 II、494. 目标和、474.一和零

01背包问题 Leetcode 1049. 最后一块石头的重量 II 题目链接&#xff1a;1049 最后一块石头的重量 II 题干&#xff1a;有一堆石头&#xff0c;用整数数组 stones 表示。其中 stones[i] 表示第 i 块石头的重量。 每一回合&#xff0c;从中选出任意两块石头&#xff0c;然后将…

网域图片的访问下载路径

网域图片的本身内容资源在网络空间中的访问下载路径

Unity(第十七部)Unity自带的角色控制器

组件Character Controller 中文角色控制器 using System.Collections; using System.Collections.Generic; using UnityEngine;public class player : MonoBehaviour {private CharacterController player;void Start(){player GetComponent<CharacterController>();}v…

Linux基本指令(上)

在Linux中&#xff0c;将文件夹称为目录&#xff0c;后面的内容都与目录相关。 1. ls指令 语法&#xff1a; ls [选项][目录或文件] 功能&#xff1a;对于目录&#xff0c;该命令列出该目录下的所有子目录与文件。对于文件&#xff0c;将列出文件名以及其他信息。 常用选项 …

Java ElasticSearch-Linux面试题

Java ElasticSearch-Linux面试题 前言1、守护线程的作用&#xff1f;2、链路追踪Skywalking用过吗&#xff1f;3、你对G1收集器了解吗&#xff1f;4、你们项目用的什么垃圾收集器&#xff1f;5、内存溢出和内存泄露的区别&#xff1f;6、什么是Spring Cloud Bus&#xff1f;7、…

Springboot解决模块化架构搭建打包错误找不到父工程

Springboot解决模块化架构搭建打包错误找不到父工程 一、情况一找不到父工程依赖1、解决办法 二、情况二子工程相互依赖提示"程序包xxx不存在" 一、情况一找不到父工程依赖 报错信息 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-deploy-plugin:…

使用Node.js构建一个简单的聊天机器人

当谈到人工智能&#xff0c;我们往往会想到什么&#xff1f;是智能语音助手、自动回复机器人等。在前端开发领域中&#xff0c;我们也可以利用Node.js来构建一个简单而有趣的聊天机器人。本文将带你一步步实现一个基于Node.js的聊天机器人&#xff0c;并了解其工作原理。 首先…

安装 Ubuntu 22.04.3 和 docker

文章目录 一、安装 Ubuntu 22.04.31. 简介2. 下载地址3. 系统安装4. 系统配置 二、安装 Docker1. 安装 docker2. 安装 docker compose3. 配置 docker 一、安装 Ubuntu 22.04.3 1. 简介 Ubuntu 22.04.3 是Linux操作系统的一个版本。LTS 版本支持周期到2032年。 系统要求双核 C…

linux c++ 开发 tensorrt 安装

tensorrt 官方下载地址&#xff08;需要注册账号登录&#xff09;&#xff1a;Log in | NVIDIA Developer 根据系统发行版和CUDA版本 (nvcc -V) 选择合适的安装包 EA&#xff08;early access&#xff09;版本代表抢先体验。 GA&#xff08;general availability&#xff09;代…

创新永不止步,织信低代码平台继续加速前进!

2023年&#xff0c;织信低代码首创“企业级”低代码概念&#xff0c;定位服务企业数字化升级战略。 经历了全年14个大版本的升级&#xff0c;下面就来细数一下2023的重大功能更新&#xff01; 1、组件设计器 织信团队凭借丰富的项目实施经验和深入客户需求理解&#xff0c;重…

vue3 构建项目

一.使用vite构建&#xff1a; npm init vitelatest 项目名称 构建的项目模板 进入项目 cd 项目名称 安装项目依赖包 npm install 启动项目 npm run dev 二.使用vue脚手架构建&#xff1a; npm init vuelatest 后续基本差不多

Win11系统安装安卓子系统教程

随着Win11系统的不断普及&#xff0c;以及硬件设备的更新换代&#xff0c;我相信很多同学都已经更新并使用到了最新的Win11系统。那么&#xff0c;Win11系统最受期待的功能“Windows Subsystem for Android”&#xff08;简称WSA&#xff09;&#xff0c;即《安卓子系统》。他可…

高马步和四平马步总结

目录 一.高马步武当袁师懋视频讲解高马步要点总结其它零散总结 二.四平马步先开胯&#xff0c;开胯的方法方法总结参考文章 三.站桩问题总结站桩时脚部灼烧感 四.记录2024 1.17 站桩突破25分钟2024年 3.1 晚上尝试四平马步&#xff0c;突破10分钟 站桩脚趾要抓地吗站桩文章 一.…

[AutoSar]BSW_Com03 DBC详解 (一)

目录 关键词平台说明一、DBC 定义1.1 相关工具 二、主要组成部分介绍2.1 Networks2.2 ECUs2.3 Network nodes2.4 messages2.5 signal2.6 Value Tables 三、主要组成部分关系图 关键词 嵌入式、C语言、autosar、OS、BSW 平台说明 项目ValueOSautosar OSautosar厂商vector &am…