数据结构基础7:二叉树【链式结构】实现和递归思想。
二叉树的链式结构实现
- 一.二叉树链式结构的实现:
- 1.前置说明:
- 1.创建二叉树:
- 2.二叉树的结构:
- 2.二叉树的遍历:
- 1.二叉树的前中后序遍历:
- 2.内容拓展:
- 二.二叉树链式(题目)
- 题目一:计算节点的个数:
- 方法一:注意事项:
- 方法二:注意事项:
- 题目二:计算叶子节点的个数:
- 方法一:
- 题目三:求第K层节点的个数:
- 方法一:
- 题目四:
- 方法一:重新定义一个函数:
- 方法二:(判断左右节点数值和root数值)
- 题目五:二叉树的最大深度:
- 方法一:
一.二叉树链式结构的实现:
1.前置说明:
对于一颗二叉树的构建是比较复杂的在刚刚开始了解二叉树的构建的时候。我们可以通过创建多个节点的方式去构建二叉树的结构,直接连接节点的左右节点构建一个二叉树方便去学习。
1.创建二叉树:
struct TreeNode* byNode(TreeNodeData x)
{struct TreeNode* tmp = (struct TreeNode*)malloc(sizeof(struct TreeNode));if (tmp == NULL){perror(tmp);exit(-1);}tmp->val = x;tmp->left = NULL;tmp->right = NULL;return tmp;
}void creatTreeNode()
{//1.构建二叉树:struct TreeNode* n1 = byNode(1);struct TreeNode* n2 = byNode(2);struct TreeNode* n3 = byNode(3);struct TreeNode* n4 = byNode(4);struct TreeNode* n5 = byNode(5);struct TreeNode* n6 = byNode(6);struct TreeNode* n7 = byNode(7);n1->left = n2;n1->right = n3;n2->left = n4;n2->right = n5;n3->left = n6;n3->right = n7;}
int main()
{//1.构建二叉树:creatTreeNode();
}
2.二叉树的结构:
1.需要注意的是上面的代码不是创建二叉树的一个正规方法,后面我的博客是会去涉及到二叉树的一个创建:
1.空树:
2.非空树:
从二叉树的概念可知二叉树是递归构建的所以后面的操作都是基于递归构建的内容。
2.二叉树的遍历:
1.二叉树的前中后序遍历:
1.学习二叉树结构,最简单的方式就是遍历。所谓二叉树遍历(Traversal)是按照某种特定的规则,依次对二叉树中的节点进行相应的操作,并且每个节点只操作一次。访问结点所做的操作依赖于具体的应用问题。 遍历是二叉树上最重要的运算之一,也是二叉树上进行其它运算的基础。
2.前中后序遍历递归结构遍历:
- 前序遍历(Preorder Traversal 亦称先序遍历)——访问根结点的操作发生在遍历其左右子树之前。
- 中序遍历(Inorder Traversal)——访问根结点的操作发生在遍历其左右子树之中(间)。
- 后序遍历(Postorder Traversal)——访问根结点的操作发生在遍历其左右子树之后。
前序遍历:
//1.前序
void PreOrder(struct TreeNode* root)
{//1.返回条件:if (root == NULL){printf("NULL ");return ;}//2.进入递归://先printf("%d ", root->val);//左PreOrder(root->left);//右PreOrder(root->right);
}
中序遍历:
//2.中序
void InOrder(struct TreeNode* root)
{//1.返回条件:if (root == NULL){printf("NULL ");return;}//2.进入递归://左PreOrder(root->left);//根:printf("%d ", root->val);//右PreOrder(root->right);
}
后序遍历:
//3.后序
void PostOrder(struct TreeNode* root)
{//1.返回条件:if (root == NULL){printf("NULL ");return;}//2.进入递归://左PreOrder(root->left);//右PreOrder(root->right);//根:printf("%d ", root->val);
}
2.内容拓展:
一.普通二叉树:
1.增删查改是没有意义的:内容上的增删查改对二叉树的结构造成破坏:二.二叉搜索树(链式结构):
1.AVL树:
2.红黑树:
3.二叉树的oj题目:
二.二叉树链式(题目)
题目一:计算节点的个数:
方法一:注意事项:
1.创建在函数里面创建静态局部变量进入递归函数这个变量不会再一次创建:
2.注意root到空的时候的返回值为0:
3.注意不要去创建多个树如果存在多个树去计算节点会导致静态变量数值的问题:
//题目一:计算节点个数://方法一(使用局部静态):
int TreeSize(TN* root)
{//1.返回条件:if (root == NULL){return 0;}//只会在第一次进入函数去定义:static int num = 0;//2.进入递归://1.当前节点:num++;//2.左:TreeSize(root->left);//3.右:TreeSize(root->right);return num;
}
方法二:注意事项:
1.使用分治的思路去每次加上一个当前节点的个数。
2.当节点为空的时候就返回一个0.
3.注意:这个函数可以同时去计算多个树的节点个数:
//方法二(使用分治的思路):
int TreeSize2(TN* root)
{if (root == NULL)return 0;//左节点+右节点return TreeSize2(root->left) + TreeSize2(root->right) + 1;
}
题目二:计算叶子节点的个数:
方法一:
1,叶子节点是没有左子树没有右子树的就是叶子节点:
2.遍历每一个节点,并且判断节点是否是叶子节点:
3.使用分治的思路去计数:
int TreeLeafSize(TN* root)
{//1.只有叶子才返回:if (root->left == NULL && root->right==NULL){return 1;}//2.进入左右子树递归:return TreeLeafSize(root->left) + TreeLeafSize(root->right);
}
题目三:求第K层节点的个数:
方法一:
1.函数需要传参数K
2.找根节点的K层就相当于找根节点左子树根的第K-1层:
3.找根节点的K层就相当于找根节点右子树根的第K-1层:
4.进入函数不要多次的进行–,k–不要写在函数中:下一次进入右树的时候就不需要再一次的–了!
//题目三:计算第K层节点个数://方法一:int TreeKSize(TN* root, int k)
{assert(k!=0);//1.如果在k层没有到的情况下到空返回0if (root == NULL){return 0;}//2.当到达K层的时候。if (k == 1){return 1;}//3.没有到达K层并且没有为空的时候就进入递归://3-1:进入不同的栈中只要是同层的就可以保证K值相同:k--;return TreeKSize(root->left, k)+TreeKSize(root->right,k);
}
题目四:
题目链接:单值二叉树
方法一:重新定义一个函数:
1.我们前面大部分的操作就是root为空就返回(true)(因为我们是比较数值是否相同所有返回true):
2…如果每一次通过root的数值去判断我们是需要传数值到递归的下一个部分:
3.如果左子树有不相等的就直接返回false(就不需要进入右子树判断):
4.如果左右相等的就继续函数不需要返回(继续进入函数)。
bool isUnivalTreeNode(struct TreeNode* root,int val)
{//1.到空树:if(root==NULL){return true;}//2.数值相同:if(root->val==val){}else if(root->val!=val){return false;}//左子树:if(isUnivalTreeNode(root->left,root->val)){//右子树:return isUnivalTreeNode(root->right,root->val);}return false;
}bool isUnivalTree(struct TreeNode* root){//左子树:if(isUnivalTreeNode(root->left,root->val)){//右子树:return isUnivalTreeNode(root->right,root->val);}return false;
}
方法二:(判断左右节点数值和root数值)
1.如果根为空就返回真:
2.如果左右子树的根节点数值和当前root的数值相同就继续递归进入。
3.如果左右子树中有一个节点数值和root的数值不相同就返回
bool isUnivalTree(struct TreeNode* root){if(root==NULL)return true;//1.这两个思路可以解决一个为空,一个有的情况。//2.解决两个都为空的情况。//3.解决两个都不是空的情况。if(root->left!=NULL && root->left->val != root->val){return false;}if(root->right!=NULL && root->right->val != root->val){return false;}//进入递归:return isUnivalTree(root->left) && isUnivalTree(root->right);
}
题目五:二叉树的最大深度:
方法一:
二叉树的深度
1.如果到空就返回0,空不是节点数。
2.每一次比较左右的数的值拿出较大的数值进行+1这个+1就相当加上当前节点。
3.子树的根节点不是空就继续。
int maxDepth(struct TreeNode* root){//1.左为空返回0(空节点不是节点数是一个返回的标志)。//2.右子树不是空就继续进入函数。if(root==NULL)return 0;//2.比较左右子树的返回值取较大的数值:int max=maxDepth(root->left);int max1=maxDepth(root->right); if(max1>max){max=max1;}//在回去的过程中自己也是节点return max+1;
}
相关文章:

数据结构基础7:二叉树【链式结构】实现和递归思想。
二叉树的链式结构实现 一.二叉树链式结构的实现:1.前置说明:1.创建二叉树:2.二叉树的结构: 2.二叉树的遍历:1.二叉树的前中后序遍历:2.内容拓展: 二.二叉树链式(题目)题目一:计算节点…...

[.NET 6] IHostedService 的呼叫等等我的爱——等待Web应用准备就绪
📢欢迎点赞 :👍 收藏 ⭐留言 📝 如有错误敬请指正,赐人玫瑰,手留余香!📢本文作者:由webmote 原创📢作者格言:新的征程,我们面对的不是技术而是人心,人心不可测,海水不可量,唯有技术,才是深沉黑夜中的一座闪烁的灯塔 !序言 在这篇文章中,我将介绍如何等…...

基于jeecg-boot的flowable流程自定义业务退回撤回或驳回到发起人后的再次流程提交
更多nbcio-boot功能请看演示系统 gitee源代码地址 后端代码: https://gitee.com/nbacheng/nbcio-boot 前端代码:https://gitee.com/nbacheng/nbcio-vue.git 在线演示(包括H5) : http://122.227.135.243:9888 主要…...

python如何学习
功能如此强大、高效的Python,却非常的简单好学,这让学它的同学爱不释手,也让越来越多的互联网企业开始用Python来做主要的开发语言,比如谷歌、Facebook(现Meta)、豆瓣、知乎等知名互联网公司都在使用Python…...
Centos7更新php7.2版本升级
之前搭建的LNMP环境php使用yum安装的版本为7.2,现有项目wordpress安装wp插件需要php7.4版本的支持,需要在原来的环境更新php版本。 一、卸载php7.2 yum remove php*原先的安装方式是yum安装直接yum remove就可以卸载否则需要rpm命令查询,按…...

操作系统学习笔记---计算机系统概述
目录 概念 功能和目标 特征 并发 共享(资源共享) 虚拟 异步 发展与分类 手工操作阶段(无OS) 批处理阶段 单道批处理系统 多道批处理系统 分时操作系统 实时操作系统 网络操作系统 分布式计算机系统 个人计算机操…...
uniapp H5 navigateBack无法返回上一层级
项目场景: 提交表单后需要返回上一级 原因分析: H5在PC端打开,当前页面重新加载的情况下,出现navigateBack不能返回,由于H5端页面刷新后返回页面栈会消失 //提交 const handleSubmit async () > {form.value?.a…...

Android性能优化之应用瘦身(APK瘦身)
关于作者:CSDN内容合伙人、技术专家, 从零开始做日活千万级APP。 专注于分享各领域原创系列文章 ,擅长java后端、移动开发、人工智能等,希望大家多多支持。 目录 一、导读二、概览2.1 apk组成 三、优化方向3.1 源代码3.1.1 代码混…...

C语言数组和指针笔试题(二)(一定要看)
目录 字符数组二例题1例题2例题3例题4例题5例题6例题7总结 字符数组三例题1例题2例题3例题4例题5例题6例题7 字符数组二 char arr[] {a,b,c,d,e,f} 1:printf("%d\n", strlen(arr)); 2:printf("%d\n", strlen(arr0)); 3:printf("%d\n", strlen(…...

uniapp——实现在线选座功能——技能提升
首先声明一点:下面的内容是从一个uniapp的程序中摘录的,并非本人所写,先做记录,以免后续遇到相似需求抓耳挠腮。 这里写目录标题 效果图代码——html部分cu-custom组件anil-seat组件 代码——jscss部分 效果图 代码——html部分 …...

领域驱动设计:微服务的各种边界
文章目录 演进式架构微服务还是小单体?微服务边界的作用 在用 DDD 进行微服务设计时,我们可以通过事件风暴来确定领域模型边界,划定微服务边界,定义业务和系统运行边界,从而保证微服务的单一职责和随需而变的架构演进能…...

MySQL之数据类型
目录 一、MySQL数据类型分类 二、数值类型 1、整数类型 2、bit类型 3、小数类型 三、字符串类型 1、char 2、varchar 3、char和varchar比较 四、日期和时间类型 五、enum和set 一、MySQL数据类型分类 MySQL 数据类型可以大致分为以下三类: 数值类型:用于…...
词法作用域改变词法作用域
一、词法作用域 1.定义: 为什么叫词法作用域?因为大部分标准语言编译器的第一个工作阶段叫作词法化,词法化的过程会对源代码中的字符进行检查,如果是有状态的解析过程,还会赋予单词语义。 简单来说࿰…...
关于C++的隐藏 (hidden),重载(overload),重写(override)小结。
关于隐藏 (hidden) 假如继承以后,子类出现父类同名函数, 即使参数的形式不同, 也会导致父类的函数隐藏, 不参与函数匹配,不能使用。 这个链接讲的不错。https://zhuanlan.zhihu.com/p/575423511 #include <iostrea…...

算法通关村18关 | 透析回溯的模板
回溯有清晰的解题模板, void backtracking(参数){if (终止条件){存放结果;return;}for (选择本层中的集合元素(画成树,就是树节点孩子的大小) {处理节点;backtracking();回溯,撤销处理结果;}} 1. 从N叉树说起 在回溯之前&#x…...
【论文阅读】Untargeted Backdoor Attack Against Object Detection(针对目标检测的无目标后门攻击)
文章目录 一.论文信息二.论文内容0.摘要1.论文概述2.背景介绍3.作者贡献4.重点图表 一.论文信息 论文题目: Untargeted Backdoor Attack Against Object Detection(针对目标检测的无目标后门攻击) 发表年份: 2023-ICASSP&#x…...

分库分表---理论
目录 一、垂直切分 1、垂直分库 2、垂直分表 3、垂直切分优缺点 二、水平切分 1、水平分库 2、水平分表 3、水平切分优缺点 三、数据分片规则 1、Hash取模分表 2、数值Range分表 3、一致性Hash算法 四、分库分表带来的问题 1、分布式事务问题 2、跨节点关联查询…...
[golang 流媒体在线直播系统] 2.搭建基于golang的流媒体服务器实现拉流推流,以及Html客户端拉取hls类型的流
一.使用 Go 语言的开源框架Livego搭建流媒体服务器 1.Livego 框架的介绍 Go 语言拥有强大的 服务器性能 ,golang 在语言级别解决了 多进程并发 的问题,支持 多核 CPU均衡使用 ,支持 海量轻量级线程 ,所以非常适合做 流媒体服务器 .而 livego 是基于golang 开发的简单高效的…...

9月12日作业
作业代码 #include <iostream>using namespace std;class Shape { protected:double cir;double area; public://无参构造Shape() {cout<<"无参构造"<<endl;}//有参构造Shape(double c, double a):cir(c), area(a){cout<<"有参构造&quo…...

React框架下如何集成H.265网页流媒体视频播放器EasyPlayer.js?
H5无插件流媒体播放器EasyPlayer属于一款高效、精炼、稳定且免费的流媒体播放器,可支持多种流媒体协议播放,可支持H.264与H.265编码格式,性能稳定、播放流畅,能支持WebSocket-FLV、HTTP-FLV,HLS(m3u8&#…...

如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)
概述 在 Swift 开发语言中,各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过,在涉及到多个子类派生于基类进行多态模拟的场景下,…...

【第二十一章 SDIO接口(SDIO)】
第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...
服务器硬防的应用场景都有哪些?
服务器硬防是指一种通过硬件设备层面的安全措施来防御服务器系统受到网络攻击的方式,避免服务器受到各种恶意攻击和网络威胁,那么,服务器硬防通常都会应用在哪些场景当中呢? 硬防服务器中一般会配备入侵检测系统和预防系统&#x…...

全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...
第25节 Node.js 断言测试
Node.js的assert模块主要用于编写程序的单元测试时使用,通过断言可以提早发现和排查出错误。 稳定性: 5 - 锁定 这个模块可用于应用的单元测试,通过 require(assert) 可以使用这个模块。 assert.fail(actual, expected, message, operator) 使用参数…...

【单片机期末】单片机系统设计
主要内容:系统状态机,系统时基,系统需求分析,系统构建,系统状态流图 一、题目要求 二、绘制系统状态流图 题目:根据上述描述绘制系统状态流图,注明状态转移条件及方向。 三、利用定时器产生时…...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...

mysql已经安装,但是通过rpm -q 没有找mysql相关的已安装包
文章目录 现象:mysql已经安装,但是通过rpm -q 没有找mysql相关的已安装包遇到 rpm 命令找不到已经安装的 MySQL 包时,可能是因为以下几个原因:1.MySQL 不是通过 RPM 包安装的2.RPM 数据库损坏3.使用了不同的包名或路径4.使用其他包…...

sipsak:SIP瑞士军刀!全参数详细教程!Kali Linux教程!
简介 sipsak 是一个面向会话初始协议 (SIP) 应用程序开发人员和管理员的小型命令行工具。它可以用于对 SIP 应用程序和设备进行一些简单的测试。 sipsak 是一款 SIP 压力和诊断实用程序。它通过 sip-uri 向服务器发送 SIP 请求,并检查收到的响应。它以以下模式之一…...