二叉树的创建与遍历

目录

前言:

二叉树的概念与结构

二叉树的链式存储

 二叉树的创建

二叉树的销毁

二叉树结点个数计算

二叉树叶子结点个数计算

二叉树第k层节点个数的计算

二叉树高度的计算

二叉树查找值为x的结点

 二叉树的遍历

二叉树的前序遍历

二叉树的中序遍历

二叉树的后序遍历

二叉树的层序遍历

判断二叉树是否为完全二叉树


前言:

二叉树的层序遍历需要使用队列实现,队列的实现见数据结构之队列,对于二叉树的代码,需要不断画递归展开图进行结合理解,递归展开图不展开描述;

二叉树的概念与结构

一棵二叉树是n个有限结点的集合,该集合或者为空,或者由一个称为根(root)的结点或者由一个根结点加上两棵别称为左子树和右子树的二叉树组成 ;

二叉树的链式存储

二叉树的链式存储结构指的是用链表表示一棵二叉树,即用链表指示元素的逻辑关系;

通常的方法是链表中每个结点由三个域组成,数据域与左右指针域,左右指针分别用来给出该结点左孩子和右孩子所在的链结点的存储地址;

 二叉树的创建

扩展二叉树是指在二叉树中出现空子树的位置增加空树叶,所形成的二叉树,使得子树成为满二叉树的二叉树叫做扩展二叉树;

先序遍历字符串 A B C # # # D # # 其中“#”表示的是空格,空格字符代表空树,则建立起来的二叉树如下图所示:

//二叉树结点结构
typedef int BTDataType;
typedef struct BinaryTreeNode
{
	struct BinaryTreeNode* left;
	struct BinaryTreeNode* right;
	BTDataType val;
}BTNode;
//创建二叉树,返回根结点地址
//A B C # # # D # #
BTNode* CreateTree(char* str, int* pi)
{
	if (str[*pi] == '#')
	{
		(*pi)++;
		return NULL;
	}

	BTNode* root = (BTNode*)malloc(sizeof(BTNode));
	root->val = str[*pi];
	(*pi)++;

	root->left = CreateTree(str, pi);
	root->right = CreateTree(str, pi);

	return root;

}
int main()
{
	char arr[100] = { 'A', 'B', 'C', '#', '#', '#', 'D', '#', '#' };
	int i = 0;
	BTNode* root = CreateTree(arr, &i);
    return 0;
}

二叉树的销毁

按照先销毁其左子树,再销毁右子树,最后销毁根的顺序递归销毁二叉树;

void DestroyTree(BTNode* root)
{
	if (root == NULL)
		return;
	DestroyTree(root->left);
	DestroyTree(root->right);
	free(root);
}

二叉树结点个数计算

若根结点为空,则二叉树结点个数=0;

若根节点不为空,则二叉树结点个数=左子树的结点个数+右子树极点个数+1;

//二叉树结点个数
int TreeSize(BTNode* root)
{
	if (root == NULL)
		return 0;
	return TreeSize(root->left) + TreeSize(root->right) + 1;
}

二叉树叶子结点个数计算

叶子结点即某一节点出的左孩子为空并且右孩子为空则为叶子结点;

//二叉树叶子结点个数
int TreeLeafSize(BTNode* root)
{
	if (root == NULL)
		return 0;
	if (root->left == NULL && root->right == NULL)
		return 1;
	return TreeLeafSize(root->left) + TreeLeafSize(root->right);
}

二叉树第k层节点个数的计算

规定:二叉树的层数K以1为起点,即第一层 k = 1;

二叉树的第k层节点个数=其左子树的第k-1层节点个数+其右子树的第k-1层节点个数;

int TreeKLevel(BTNode* root,int k)
{
	assert(k > 0);
	if (root == NULL)
		return 0;
	if (k == 1)
		return 1;
	return TreeKLevel(root->left, k - 1) + TreeKLevel(root->right, k - 1);
}

二叉树高度的计算

二叉树的高度=左子树的高度与右子树的高度较大值+1;

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

二叉树查找值为x的结点

如根节点非空,先查找根,如果根结点处的值即为查找的值,直接返回根节点的地址,如果根节点处未查找到,先查找左子树,找到了直接返回,找不到继续查找右子树;

//二叉树查找值为x的结点
BTNode* TreeFind(BTNode* root, BTDataType x)
{
	if (root == NULL)
		return NULL;
	if (root->val == x)
		return root;
	struct BinaryTreeNode* ret = NULL;
	ret = TreeFind(root->left, x);
	if (ret != NULL)
		return ret;
	ret = TreeFind(root->right, x);
	if (ret != NULL)
		return ret;
	return NULL;
}

 二叉树的遍历

  • 前序遍历NLR(依次遍历:根结点—左子树—右子树)
  • 中序遍历LNR(依次遍历:左子树—根结点—右子树)
  • 后序遍历LRN(依次遍历:左子树—右子树—根结点)

 由于二叉树是递归定义的,按照下图所示方法,便可得到前序遍历的结点顺序

 前序遍历结果:A->B->D->E->C->F->G        

中序遍历结果为D->B->E->A->F->C->G;后序遍历结果:D->E->B->F->G->C->A;

二叉树的前序遍历

//二叉树的前序遍历
void PrevOrder(BTNode* root)
{
	if (root == NULL)
		return;
	printf("%c ", root->val);
	PrevOrder(root->left);
	PrevOrder(root->right);
}

二叉树的中序遍历

//二叉树的中序遍历
void InOrder(BTNode* root)
{
	if (root == NULL)
		return;
	InOrder(root->left);
	printf("%c ", root->val);
	InOrder(root->right);
}

二叉树的后序遍历

//二叉树的后序遍历
void PostOrder(BTNode* root)
{
	if (root == NULL)
		return;
	PostOrder(root->left);
	PostOrder(root->right);
	printf("%c ", root->val);
}

二叉树的层序遍历

二叉树的层序遍历是指将二叉树从上到下,从左到右依次遍历,如下图所示层序遍历结果为

A—>B—>D—>C

层序遍历的思路如下:

1.  判断根结点是否为空,非空则入队列

2. 判断队列是否为空,不为空则队头元素出队列,再将队头元素的孩子入队列,若队头元素的孩子为空,则不入队列;

3.判断队列是否为空,将B出队列,再将B的孩子入队列;

4.判断队列是否为空,将D出队列,再将D的孩子入队列;

5. 判断队列是否为空,将C出队列,再将C的孩子入队列;队列为空时结束遍历;

//二叉树的层序遍历
void BinaryTreeLevelOrder(BTNode* root)
{
	Queue q;
	InitQueue(&q);
	if (root != NULL)
		QueuePush(&q, root);
	while (!QueueEmpty(&q))
	{
		BTNode* front = QueueFront(&q);
		printf("%c ", front->val);
		if (front->left != NULL);
		    QueuePush(&q, front->left);

		if (front->right != NULL)
			QueuePush(&q, front->right);

		QueuePop(&q);
	}
	DestroyQueue(&q);
}

判断二叉树是否为完全二叉树

满二叉树:当二叉树深度为K,二叉树结点个数为2^k-1,则为满二叉树;

对满二叉树结点位置进行编号

  • 编号的规则:自上而下,从左向右 ;
  • 每一节点位置都有元素;

完全二叉树

深度为K具有n个结点的二叉树,当且仅当其每一个节点都与深度为k的满二叉树中的编号为1~n的节点一 一对应时,称为完全二叉树;

判断二叉树是否是完全二叉树,是完全二叉树返回true,否则返回false;
利用二叉树层序遍历,非空结点连续为完全二叉树,非空结点不连续,存在空节点,则为非完全二叉树;

int BinaryTreeComplete(BTNode* root)
{
	Queue q;
	InitQueue(&q);
	if (root != NULL)
		QueuePush(&q, root);
	while (!QueueEmpty(&q))
	{
		BTNode* front = QueueFront(&q);
		if (front == NULL)
			break;
		QueuePush(&q, front->left);
		QueuePush(&q, front->right);
		QueuePop(&q);
	}
	//若已经出现空结点,队列后面结点存在非空结点,则为非完全二叉树
	while (!QueueEmpty(&q))
	{
		BTNode* front = QueueFront(&q);
		QueuePop(&q);

		if (front != NULL)
		{
			DestroyQueue(&q);
			return false;
		}
	}
	DestroyQueue(&q);
	return true;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

还在为忘记BIOS密码担心?至少有五种方法可以重置或删除BIOS密码

忘记密码是一个我们都非常熟悉的问题。虽然在大多数情况下,只需单击“忘记密码”选项,然后按照几个简单的步骤即可恢复访问权限,但情况并非总是如此。忘记BIOS密码(通常为避免进入BIOS设置或避免个人计算机启动而设置的密码)意味着你将无法完全启动系统。 幸运的是,就像…

oracle初步学习

先了解几个登录的方式 1.直接在cmd输入 sqlplus scott/tiger //登陆普通用户scott 2.输入sqlplus sys as sysdba 之后 紧接着让你输入口令,直接输入sys就行了 3.先输入sqlplus/nolog 在输入conn system/managerorcl 先在cmd窗口输入sqlplus/nolog &#x…

免费分享一套基于Springboot+Vue的医院管理系统,挺漂亮的

大家好,我是java1234_小锋老师,看到一个不错的SpringbootVue的医院管理系统,分享下哈。 项目视频演示 【免费】springbootvue医院管理系统 Java毕业设计_哔哩哔哩_bilibili【免费】springbootvue医院管理系统 Java毕业设计项目来自互联网&a…

使用IDEA 将Eclipse java工程转为maven格式

使用IDEA 将Eclipse java工程转为maven格式 ①使用idea打开项目,在项目根目录下右键选择 Add Framework Support 选择 maven ,引入maven ②找到项目中的.classpath文件或者lib目录 根据.classpath文件或者lib目录中列举的jar包名,将其依次手…

【架构师】的修炼之道都需要学习哪些?看看这些就够了

👨‍🎓博主简介 🏅云计算领域优质创作者   🏅华为云开发者社区专家博主   🏅阿里云开发者社区专家博主 💊交流社区:运维交流社区 欢迎大家的加入! 🐋 希望大家多多支…

React父组件怎么调用子组件的方法

调用方法:1、类组件中的调用可以利用React.createRef()、ref的函数式声明或props自定义onRef属性来实现;2、函数组件、Hook组件中的调用可以利用useImperativeHandle或forwardRef抛出子组件ref来实现。 【程序员必备开发工具推荐】Apifox一款免费API管理…

nginx关闭重启和配置检查

优雅关闭Nginx 找出nginx的进程号:ps -ef | grep nginx 执行命令:kill -QUIT 主pid 注意: 其中pid是主进程号的pid(master process),其他为子进程pid(worker process) 这种关闭方式…

计算机视觉基础(9)——相机标定与对极几何

前言 本节我们将学习相机标定和对极几何两部分的内容。 在相机标定部分,我们将学习直接线性变换(Direct Linear Transform, DL),张正友标定法(Zhang’s Method)和 Perspective-n-Point (PnP) 这三种方法。 在对极几何部…

赢麻了……腾讯1面核心9问,小伙伴过了提42W offer

说在前面 在40岁老架构师尼恩的(50)读者社群中,经常有小伙伴,需要面试腾讯、美团、京东、阿里、 百度、头条等大厂。 下面是一个小伙伴成功拿到通过了腾讯面试,并且最终拿到offer,一毕业就年薪42W&#x…

rabbit的扇出模式(fanout发布订阅)的生产者与消费者使用案例

扇出模式 fanout 发布订阅模式 生产者 生产者发送消息到交换机(logs),控制台输入消息作为生产者的消息发送 package com.esint.rabbitmq.work03;import com.esint.rabbitmq.RabbitMQUtils; import com.rabbitmq.client.Channel;import java.util.Scanne…

csapp第三章读书笔记

caspp chapter 3 寄存器 operand form data movement instructions mov 指令例子: 0扩展 movz 指令: Zero-extending data movement instructions是一种计算机指令类型,涉及将数据从一个位置移动到另一个位置,同时通过在最重要的一端添加零位来将数据扩…

精美可视化:Python自动化生成漂亮的测试报告

“ 运用Python的Unittest、数据驱动测试(DDT)、Excel、Jinja2和HTML技术,构建一个能够自动生成精美可视化测试报告的自动化测试框架” 思路流程 封装读取数据,让所有数据都能够再excel中填写,不再填写任何一行逻辑代码…

【前端】使用json-server报错

当我们使用json-server模仿后端接口时需要运行json-server --watch index.json这个命令生成增删改查接口但是可能会报这个错误,如图 这时我们运行 npm i json-server -g命令即可,然后再重新运行json-server --watch index.json就行了

编码器脉冲信号测量2路DI高速计数器PNP/NPN转RS-485数据采集模块 解码转换成标准Modbus RTU协议 YL150-485

特点: ● 编码器解码转换成标准Modbus RTU协议 ● 可用作编码器计数器或者转速测量 ● 支持编码器计数,可识别正反转 ● 也可以设置作为2路独立DI高速计数器 ● 计数值支持断电自动保存 ● DI输入支持PNP和NPN输入 ● 继电器和机械开关输入时可以…

C++入门(2)—函数重载、引用

目录 一、函数重载 1、参数类型不同 2、参数个数不同 3、参数顺序不同 4、 链接中如何区分函数重载 二、引用 1、规则 2、特征 3、使用场景 做参数 做返回值 4、常引用 5、传值、传引用效率比较 6、引用和指针的区别 接上一小节C入门(1)—命名空间、缺省参数 一…

解决requests库中的期限处理问题:从404到异常再到修复

在使用requests库进行网络请求时,用户可能会遇到一个奇怪的问题:当没有指定请求的期限时,他们得到的响应是404错误,但是一旦指定了请求的期限,就立刻遇到了一个异常,声称远程主机强制关闭了连接。这个问题让…

pytorch文本分类(一):文本预处理

pytorch文本分类(一):文本预处理 本文为自己在鲸训练营答题总结,作业练习都在和鲸社区数据分析协作平台 ModelWhale 上。 🚩学习任务原链接在这里 相关数据链接:https://pan.baidu.com/s/1iwE3LdRv3uAkGGI…

Spring Boot使用EhCache完成一个缓存集群

在上一篇在SpringBoot中使用EhCache缓存,我们完成了在Spring Boot中完成了对EhCaChe的使用,这篇,我们将对EhCache的进一步了解,也就是搭建一个EhCache的缓存集群。 集群 在搭建一个EhCache的时候,我们需要先了解&…

【linux】nmon 工具使用

nmon 介绍 nmon是奈杰尔的性能监视器的缩写,适用于POWER、x86、x86_64、Mainframe和现在的ARM(Raspberry Pi)上的Linux。同样适用于nmon for AIX的工具(与IBM的AIX一起提供)。njmon与之类似,但将数据保存为…

分类预测 | Matlab实现PSO-GRU-Attention粒子群算法优化门控循环单元融合注意力机制多特征分类预测

分类预测 | Matlab实现PSO-GRU-Attention粒子群算法优化门控循环单元融合注意力机制多特征分类预测 目录 分类预测 | Matlab实现PSO-GRU-Attention粒子群算法优化门控循环单元融合注意力机制多特征分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现PSO…