数据结构之链式二叉树

当我们初步了解二叉树后

我们就可以进一步去深入学习二叉树了

1.链式二叉树的遍历

这里我们先去定义链式二叉树的结构

分为两个指针

一左一右

他们分别指向左子树和右子树

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

左右子树的节点又可以细分为:根,左子树,右子树

图中的1节点就是根,2和3则是左子树,456则是右子树

1.1二叉树的前序,中序,后序遍历


前序遍历、中序遍历和后序遍历。这些遍历方式指的是节点访问的顺序。

前序遍历:在前序遍历中,我们首先访问根节点,然后递归地进行左子树的前序遍历,接着递归地进行右子树的前序遍历。


中序遍历:中序遍历中,我们首先递归地进行左子树的中序遍历,然后访问根节点,最后递归地进行右子树的中序遍历。


后序遍历:后序遍历中,我们首先递归地进行左子树的后序遍历,然后递归地进行右子树的后序遍历,最后访问根节点。

而这里呢我们又遇到了老朋友递归

问题不大

我们利用一棵树为例子

图中的这一棵树

以一为根

接下来我们先进行前序遍历

1.1.1前序遍历

前序遍历中先根后左右

●所以我们先去访问根1

●访问完根1后就访问左节点2

●接下来以2为根访问2的左节点3

●然后以3为根访问3的左节点,这是他的左节点为空,所以我们返回也就是开始进行递归

●递归到一后,开始进行访问1的右节点4

●访问到四就以4为根访问4的左节点5

●访问5后发现没有左节点就递归到4访问4的右节点

●访问6时没有左节点右节点就开始递归到4再到1

●最后访问结束

这里呢,我们用N代表空

那么访问完打印后应该是这样的

1 2 3 N N N 4 5 N N 6 N N

1.1.2中序遍历

讨论完前序遍历我们进行中序遍历

中序遍历讲的是先左后根最后右

还是利用这棵树

遍历顺序:

●先是访问左子树,1的左子树是2,2的则是3,3没有了左子树也就是空,返回3,然后在访问3的右子树,空,返回到2

这是返回的应该是

N 3 N

●回到2后访问2的左子树空,所以返回到1

N 3 N 2 N

●回到一就访问1的右子树4,然后到4的左子树5,在访问5的左子树空,这时返回到5

N 3 N 2 N 1 N 5 N

这时会有疑问,为什么没有四,不是先访问到4吗?

这里没有4的原因是,返回到1后应该访问它的右子树,而右子树中还有一个左子树5,所以应该先访问5,这里的5优先访问

●然后返回到4,访问4的右子树6,这里优先访问6的左子树,为空,返回到6,访问右子树,为空

N 3 N 2 N 1 N 5 N 4 N 6 N

1.1.3后序遍历

后序遍历则是先左后右最后根

遍历顺序:

还是以这棵树为例


●先访问1的左子树,到3时,他的左子树为空,右子树为空

N N 3

●返回到2后,右子树为空,访问根2,返回1

N N 3 N 2

●回到一后,访问一的右子树,同时优先访问右子树中的左子树也就是节点五,访问五的左子树,然后是右子树,然后是五

 N N 3 N 2 N N 5

●J返回到五后访问4的右子树6,然后访问6的左子树和右子树

N N 3 N 2 N N 5 N N 6

●访问完6后就返回访问4,然后访问1

N N 3 N 2 N N 5 N N 6 4 1

2.遍历代码的实现

2.1.树的实现

首先我们需要手搓一棵树

定义出来树的结构

并需要创建新的空间,所以这里包装一个函数

typedef struct BinTreeNode
{struct BinTreeNode* left;struct BinTreeNode* right;int val;
}BTNode;
BTNode* BuyBTNode(int val)
{BTNode* newnode = (BTNode*)malloc(sizeof(BTNode));if (newnode == NULL){perror("malloc fail");return NULL;}newnode->val = val;newnode->left = NULL;newnode->right = NULL;return newnode;
}
// 手搓一棵树
BTNode* CreateTree()
{BTNode* n1 = BuyBTNode(1);BTNode* n2 = BuyBTNode(2);BTNode* n3 = BuyBTNode(3);BTNode* n4 = BuyBTNode(4);BTNode* n5 = BuyBTNode(5);BTNode* n6 = BuyBTNode(6);n1->left = n2;n1->right = n4;n2->left = n3;n4->left = n5;n4->right = n6;return n1;
}

这样就形成了图形中的树

2.2前序遍历代码

前序遍历的话我们需要用到递归

首如果检测到节点为空,这样的话就打印N并返回

如果没有

那么递归继续往下

因为是前序遍历

所以先递归左然后再递归右

void PreOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}printf("%d ", root->val);PreOrder(root->left);PreOrder(root->right);
}

2.3中序遍历代码

中序遍历和前序遍历的不同是前序遍历先根后左右,中序遍历则是先左后根最后右

所以,我们还是先遇到空返回N

没有则是返回左,打印根ra

void InOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}InOrder(root->left);printf("%d ", root->val);InOrder(root->right);
}

2.4后序遍历代码

后序遍历代码则是先左右最后根

所以

void PostOrder(BTNode* root)
{if (root = NULL){printf("N");return;}PostOrder(root->left);PostOrder(root->right);printf("%d ", root->val);
}

2.5测试的结果

3.获取节点个数

获取节点个数,正常情况下我们的思路是定义一个size

然后在遍历的时候进行++size

代码如下

int TreeSize(BTNode* root)
{static int size = 0;if (root == NULL)return 0;else++size;TreeSize(root->left);TreeSize(root->right);return size;
}

但这样会有一个缺点

我们没法去在这个函数里面重置我们的size

所以我们需要再主函数中

每调用完TreeSize函数,就需要重置一遍size

所以我们还有另外一个思路

直接去返回它的左节点和右节点,最后加一

利用递归的思想

代码如下

int TreeSize(BTNode* root)
{return root == NULL ? 0 :TreeSize(root->left) + TreeSize(root->right) + 1;
}

这样就非常巧妙的完成了节点的个数

1.获取叶节点个数
获取叶子结点个数,我们这里也用递归的方法

利用分治思想去解决这个问题

●代码思想:

1. 当遇到空树或者遇到空的节点时,也就是说这是的叶子为NULL,这是我们返回0

2. 当遇到左节点或者右节点为空,当节点不为空时,此时已经到达了叶子结点,所以返回1

3. 当遇到的不是叶节点时,我们需要到递归左节点的个数和右节点的个数,并进行递归返回

●代码思想:

对于整棵树来说,当我们遇到空树或者遇到节点为空的时候,这时的叶子结点为空,我们这时返回0,当不是上中情况的时候,我们从根往下去搜索,先搜索左节点,当左节点不为空,并且左节点的左子树和右子树都是空的时候,这时候就可以确定它是叶子了,也就是返回1,当搜索完左子树就可以搜索右子树,右子树也同理 

4.获取树的高度 

获取树的高度,我们也是利用分治的思想去实现这个代码

首先就是当我们要想返回高度的时候,我们需要调用到左右子树的高度

然后比较左右子树的高度,比较出最大的一个并返回

然后加1(因为我们递归的是左右子树的高度,我们需要整个树的高度,所以还需要加上根,也就是加一)

●代码思想:

1.当我们遇到空树或者遇到的节点为NULL,这时返回0

2.然后接下来去递归左子树和右子树

3.返回时,如果左子树大于右子树,那么就是左子树高度+1,否则右子树高度+1

 

//获取树的高度
int TreeHeight(BTNode* root)
{if (root == NULL){return 0;}TreeHeight(root->left);TreeHeight(root->right);return (TreeHeight(root->left) > TreeHeight(root->right) ? TreeHeight(root->left) : TreeHeight(root->right)) + 1;
}

但这个代码有一定的缺陷

我们可以看到,这个代码我们调用了两次TreeHeight(root->left)和TreeHeight(root->right)

在这一树中,我们调用多次函数,大大增加了计算的难度,在一棵小树中可能不明显,可当树更大时,这时候弊端就先显示出来了

所以我们可以改进一下代码,定义两个变量去接受返回值

然后比较两个返回值

//改进代码
int TreeHeight(BTNode* root)
{if (root == NULL){return 0;}/*TreeHeight(root->left);TreeHeight(root->right);return (TreeHeight(root->left) > TreeHeight(root->right) ? TreeHeight(root->left) : TreeHeight(root->right)) + 1;*/int Heightleft = TreeHeight(root->left);int Heightright = TreeHeight(root->right);return (Heightleft > Heightright ? Heightleft : Heightright + 1);
}

 

5.计算第K层节点个数 

计算k层节点的个数,我们可以看成计算左节点的(k-1)层和右节点(k-1)层的节点个数

因为我们不算顶部节点所以应该是k-1

●代码思想:

首先是如果是空树或者当遇到叶子结点外的空节点时,返回0

当遇到k为1的时候,这时只有一个根,也就返回1

其余情况均利用递归思想,去递归左右子树,注意此时的k应该变成k-1

//计算树k层的节点个数
int TreeKCount(BTNode* root, int k)
{if (root == NULL || k < 1){return 0;}if (k == 1){return 1;}return TreeKCount(root->left, k - 1) + TreeKCount(root->right, k - 1);
}

 

6.寻找某个节点 

寻找某个节点的话,我们也利用递归的方法,分治的思想去解决这个问题

寻找某个节点,那么这个节点如果不在根上,那么就在根的左子树和右子树上

那么就想下寻找

下边的节点也可以分为左子树和右子树和根

依次进行,就形成了递归

//寻找某个节点
BTNode* TreeFind(BTNode* root, int x)
{if (root == NULL){return 0;}if (x == root->val){return root;}TreeFind(root->left, x);TreeFind(root->right, x);return NULL;
}

很多人可能会想到这样的代码

可当我们去运行的时候,我们会发现找不到,不管x为多少都找不到

为什么呢?

原因是我们没有东西去接收

当我们找到的时候,我们递归需要往上递归

可上边的栈中没有可以接受的变量值

所以我们最终遍历完整棵树也找不到我们想找的节点

所以改一下代码

//寻找某个节点
BTNode* TreeFind(BTNode* root, int x)
{if (root == NULL){return 0;}if (x == root->val){return root;}BTNode* ret1 = TreeFind(root->left, x);BTNode* ret2 = TreeFind(root->right, x);if (ret1)return ret1;if (ret2)return ret2;return NULL;
}

 

这样我们利用新建立的节点去接受我们的左右子树的数据

然后如果不为空就不断返回,为空那么就返回0

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

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

相关文章

【设计模式】一、设计模式概述

文章目录 一、设计模式概述&#xff08;一&#xff09;设计模式是什么1. 设计模式的定义2. 设计模式的组成要素3、常用设计模式一览表 &#xff08;二&#xff09;设计模式的优点&#xff08;用途&#xff09;※ 本文小结 一、设计模式概述 &#xff08;一&#xff09;设计模式…

tcp/ip协议2实现的插图,数据结构8 (30 - 32章)

(201) 201 三十0 中断优先级补充 (202) 202 三十1 TCP的用户需求 函tcp_usrreq一 (203) 203 三十2 TCP的用户需求 函tcp_usrreq二 (204) 204 三十3 TCP的用户需求 函tcp_usrreq三 (205) 205 三十4 TCP的用户需求 函tcp_usrreq四 (206) 206 三十5 TCP的用户需求 函tcp_usrreq五 …

docker实战(3)

1, jdk17版本的dockerfile FROM eclipse-temurin:17-jre ADD ./target/ods-poc-0.0.1-SNAPSHOT.jar /home/app/app.jar WORKDIR /home/app ENV TZ Asia/Shanghai ENTRYPOINT ["java","-jar","/home/app/app.jar"] EXPOSE 8083dockerfile 1, …

emment语法

文章目录 1. 生成普通的标签2. 生成 div类名3. 生成指定标签类名/id 值4. 生成带有子元素的标签5. 生成内部文本6. 一次可以生成多个标签7. 生成带有指定属性 的元素8. 生成相邻兄弟元素 1. 生成普通的标签 本质使用的就是元素选择器&#xff0c;例如 div p a 标签等等。 2. …

Openlayers入门教程 --- 万字长篇

也许你还不熟悉Openlayers&#xff0c;也许你是一个Openlayers小白&#xff0c;零基础没关系&#xff0c;这篇文章提供最基础的 Openlayers 教程&#xff0c;简单易学&#xff0c;贯穿整个Openlayers 知识体系。读完本文&#xff0c;您将会对 Openlayers 有一个全新的认识。 文…

图像分类技术在电商平台的创新应用与实践

一、引言 在当今快速发展的互联网电商领域&#xff0c;商家面临着激烈的竞争和不断变化的市场需求。我们在服务电商的过程中&#xff0c;利用AI大模型技术创新性地引入了图像分类技术&#xff0c;为供应链管理带来了革命性的变革。接下来&#xff0c;我们将深入探讨这一项目的…

51单片机基础篇系列-超声波测距

&#x1f308;个人主页&#xff1a;会编辑的果子君 &#x1f4ab;个人格言:“成为自己未来的主人~” HC-SR04产品特点 典型工作用电压&#xff1a;5V 超小静态工作电流&#xff1a;小于2mA 感应角度&#xff1a;不大于15度 探测距离&#xff1a;2cm-400cm 高精度&#…

论文阅读:LAPFormer: A Light and Accurate PolypSegmentation Transformer

这是一个基于Transformer的轻量级图像分割模型。作者们使用MiT&#xff08;Mix Transformer&#xff09;作为编码器&#xff0c;并为LAPFormer设计了一个新颖的解码器&#xff0c;该解码器利用多尺度特征&#xff0c;并包含特征精炼模块和特征选择模块&#xff0c;以生成精细的…

16. UE5 RPG获取GE应用的回调,并根据Tag设置数据显示到窗口

在上一篇介绍了对标签如何在项目中设置&#xff0c;这一篇先讲解一下如何在GE里面使用GameplayTag标签。 之前我在第十一章节中 11. UE5 RPG使用GameplayEffect修改角色属性&#xff08;二&#xff09;介绍了一些GE的属性&#xff0c;在UE 5.3版本中&#xff0c;修改的配置方式…

药店在线客服机器人的设计与开发

摘要 随着信息技术的快速发展和人工智能领域的不断进步&#xff0c;药店在线客服机器人已经成为现代药店服务模式中不可或缺的一部分&#xff0c;这种机器人技术通过模拟人类客服代表&#xff0c;能够提供24/7的咨询服务&#xff0c;从而大幅提高服务效率和顾客满意度&#xff…

集合系列(二) -List接口详解

一、List简介 List 的数据结构就是一个序列&#xff0c;存储内容时直接在内存中开辟一块连续的空间&#xff0c;然后将空间地址与索引对应。 以下是List集合简易架构图 由图中的继承关系&#xff0c;可以知道&#xff0c;ArrayList、LinkedList、Vector、Stack都是List的四个…

软件设计和体系结构

软件设计和体系结构 一、引言 软件 定义&#xff1a;一系列按照特定顺序组织的计算机数据、指令的集合 特点&#xff1a; 软件不是生产制造&#xff0c;是设计开发软件不会磨损和老化软件需要根据实际情况进行定制开发 软件设计的基本原则 抽象方法 过程抽象&#xff1a;是指…

进程和ELF文件

程序的二进制格式&#xff1a; 关联知识&#xff1a;GCC编译原理与使用-CSDN博客 linux二进制程序的严格格式&#xff1a;ELF&#xff08;Executeable and Linkable format&#xff09; 一 预处理&#xff0c;编译&#xff0c;汇编得到的.o文件&#xff0c;就是ELF的第一种类…

Altium Designer快速入门及项目实战教程之层次原理图PCB设计(七)

一、简介 层次PCB设计是电子工程领域中一项至关重要的任务&#xff0c;它涉及到电路板的各个层面的功能分配和布局规划。这一设计过程的关键在于如何合理安排每一层的功能和布局&#xff0c;以确保电路板的性能达到最佳状态&#xff0c;并同时保证其可靠性。 首先&#xff0c…

数据结构 第3章:栈与队列

文章目录 1. 栈1.1 栈的基本概念1.2 栈的基本操作1.3 栈的顺序存储实现1.4 栈的链式存储实现 2. 队列2.1 队列的基本概念2.2 队列的基本操作2.3. 队列的顺序存储实现2.4 队列的链式存储实现2.5 双端队列 3. 栈与队列的应用3.1 栈在括号匹配中的应用3.2 栈在表达式求值中的应用3…

6.JavaWebJDBC连接池、JDBCTemplate

目录 导语&#xff1a; 一、JDBC连接池概述 常见JDBC连接池&#xff1a; JDBC连接池的优点 二、JDBCTemplate介绍 JDBCTemplate的核心组件 JDBCTemplate的使用方法 JDBCTemplate的优点 结语&#xff1a; ​​​​​​​ 导语&#xff1a; 在Java数据库编程中&#xf…

macOS Monterey 12.7.4 (21H1123) Boot ISO 原版可引导镜像下载

macOS Monterey 12.7.4 (21H1123) Boot ISO 原版可引导镜像下载 3 月 8 日凌晨&#xff0c;macOS Sonoma 14.4 发布&#xff0c;同时带来了 macOS Ventru 13.6.5 和 macOS Monterey 12.7.4 安全更新。 本站下载的 macOS 软件包&#xff0c;既可以拖拽到 Applications&#xf…

Hive-技术补充-初识ANTLR

一、背景 要清晰的理解一条Hql是如何编译成MapReduce任务的&#xff0c;就必须要学习ANTLR。下面是ANTLR的官方网址&#xff0c;下面让我们一起来跟着官网学习吧&#xff0c;在学习的过程中我参考了《antlr4权威指南》&#xff0c;你也可以读下这本书&#xff0c;一定会对你有…

在idea中配置tomcat服务器,部署一个项目(下载教程加链接)

第一步&#xff1a;把Tomcat下载好 ww​​​​​​​Apache Tomcat - Welcome! 链接如上&#xff1a;进去后在左边找到Tomcat8点击进去后 找到图下内容 第二步&#xff1a; 打开这个文件点击bin进去 会出现一个黑色框框&#xff0c;也就是服务器 完成后就可以在浏览器输入…

【C++】实现红黑树

目录 一、认识红黑树1.1 概念1.2 定义 二、实现红黑树2.1 插入2.2 与AVL树对比 一、认识红黑树 1.1 概念 红黑树是一个二叉搜索树&#xff0c;与AVL树相比&#xff0c;红黑树不再使用平衡因子来控制树的左右子树高度差&#xff0c;而是用颜色来控制平衡&#xff0c;颜色为红色…