【数据结构和算法】--- 二叉树(4)--二叉树链式结构的实现(2)

目录

  • 一、二叉树剩余函数
    • 1.1二叉树的层序遍历
    • 1.2判断二叉树是否为完全二叉树
    • 1.3二叉树销毁
  • 二、二叉树的构建及遍历OJ题

一、二叉树剩余函数

1.1二叉树的层序遍历

层序遍历: 除了先序遍历、中序遍历、后序遍历外,还可以对二叉树进行层序遍历。设二叉树的根节点所在层数为1,层序遍历就是从所在二叉树的根节点出发,首先访问第一层的树根节点,然后从左到右访问第2层上的节点,接着是第三层的节点,以此类推,自上而下,自左至右逐层访问树的结点的过程就是层序遍历。 可以参考下图:
在这里插入图片描述
由上图不难看出,我们要控制函数一层一层的遍历二叉树。且普通递归都是会先遍历整个左子树,然后才会进入右子树。这里可以引用队列(先进先出)的概念,思路如下:
先将根节点地址入队列,然后利用循环,每当有节点出队列时就将此节点的左孩子(root -> left)的地址和右孩子(root -> right)的地址入队列(当然入队列的节点不为空,需要判断一下)。当队列为空时(while(!QueueEmpty(&q))),就结束循环并销毁队列。主要还是利用队列的思想,只要想到就不难,代码如下:

//二叉树层序遍历
void BinaryTreeLevelOrder(TreeNode* root)
{
	Queue q;
	QueueInit(&q);
	//根节点入队列
	if (root)
		QueuePush(&q, root);
	while (!QueueEmpty(&q))
	{
	    //出队列
		TreeNode* front = QueueFront(&q);
		QueuePop(&q);
		printf("%d ", front->val);
		//左右孩子判空,并入队列
		if (root->left)
			QueuePush(&q, root->left);
		if (root->right)
			QueuePush(&q, root->right);
	}
	putchar('\n');
	//销毁
    QueueDestroy(&q);
}

代码图解:
在这里插入图片描述

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

这里先要介绍一下两种特殊的二叉树:

  1. 满二叉树: 一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是说,如果一个二叉树的层数为K,且结点总数是2^k-1 则它就是满二叉树。
  2. 完全二叉树: 完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树而引出来的。对于深度为K的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从1至n的结点一一对应时称之为完全二叉树。 要注意的是满二叉树是一种特殊的完全二叉树。(即如果一棵树只有右节点没有左节点,那不能成为完全二叉树)。

根据上面的介绍,不难发现如果想要判断二叉树是否为完全二叉树,还是要一层一层的遍历二叉树。 既然如此,那么此函数还是可以使用队列来实现。代码设计思路:
第一步: 同样可以先创建队列并初始化,将第一个根节点的地址先入队列。同样执行循环,将一个节点出队列时,并将此节点的左孩子地址(root->left)和右孩子地址(root->right)都入队列。不同的是如果遇到空节点(无左孩子或右孩子便是NULL)同样要进入队列,并以队列为空(while (!QueueEmpty(&q)))作为循环结束条件(事实上此循环无法通过此条件结束)。在循环内部,如果接收到的出队列的节点为空,同样结束循环(break)。
至于遇到空节点,为什么要结束循环?因为我们判断的是完全二叉树,在进行层序遍历时,不会出现两个有效节点间还有一个空节点的情况(可以参考完全二叉树结构来思考!)。
第二步: 进行队列后续节点判断,同样可以依靠循环来不断出队列节点,当队列所出的节点不是空时(front != NULL)就直接销毁队列,并返回false;如果队列遍历到最后都没有发现空节点,那么便结束循环,销毁队列并返回true

代码如下:

//判断树是否为完全二叉树
bool BinaryTreeComplete(TreeNode* root)
{
	Queue q;
	QueueInit(&q);
	//根节点入队列
	if (root)
		QueuePush(&q, root);
	while (!QueueEmpty(&q))
	{
	    //队首出队列
		TreeNode* front = QueueFront(&q);
		QueuePop(&q);
		if (front == NULL)
			break;
		//左右孩子入队列
		QueuePush(&q, root->left);
		QueuePush(&q, root->right);

	}
	//队列遇空,判断后续节点
	while (!QueueEmpty(&q))
	{
		TreeNode* front = QueueFront(&q);
		QueuePop(&q);
		//当还有非空节点时说明二叉树为非完全二叉树
		//销毁并返回false
		if (front != NULL)
		{
			QueueDestroy(&q);
			return false;
		}
	}
	//直到队列为空,认为找到非空节点
	//销毁队列并返回true
	QueueDestroy(&q);
	return true;
}

代码图解:
在这里插入图片描述
有关层序遍历的例题:

  1. 某二叉树的后序遍历序列与中序遍历序列相同,均为 ABCDEF ,则按层次输出(同一层从左到右)的序列(A

    A FEDCBA
    B CBAFED
    C DEFCBA
    D ABCDEF

解: 解此题首先要搞清,后序遍历和中序遍历的特性,(后序:左子树,右子树,根;中序:左子树,根,右子树),根据题目所描述的后序和中序遍历相同,那么这棵二叉树必定没有右子树。
既然这样,那么可以依据后序遍历确定根节点为F,然后依次判断以后节点。

1.3二叉树销毁

关于二叉树的销毁,可以使用后续遍历的思想。 因为如果先释放上层节点,那么下层的节点将无法寻到。如:free(root);,那么root便会变为野指针,root->left来找寻左子树也是非法的。

//二叉树的销毁
void BinaryDestroy(TreeNode* root)
{
	if (!root)
		return;
	BinaryTreeDesTroy(root->left);
	BinaryTreeDesTroy(root->right);
	free(root);
}

二、二叉树的构建及遍历OJ题

二叉树的构建及遍历,牛客刷题:KY11 二叉树遍历

题目描述: 编一个程序,读入用户输入的一串先序遍历字符串,根据此字符串建立一个二叉树(以指针方式存储)。 例如如下的先序遍历字符串ABC##DE#G##F###其中“#”表示的是空格,空格字符代表空树。建立起此二叉树以后,再对二叉树进行中序遍历,输出遍历结果。

解此题首先要创建二叉树的节点(值val,指向左孩子指针left,指向右孩子指针right)。然后根据字符串arr中的数据来,连接各个节点,需要注意的是要使用前序遍历来连接。至于为什么要传pi的地址? 因为在递归过程中需要取到字符串arr不同下标位置的字符,传地址,在递归过程便可任意改变pi的值。

代码如下:

#include <stdio.h>
#include<stdlib.h>

typedef char BTDataType;
//二叉树节点
typedef struct BinaryTree
{
    BTDataType val;
    struct BinaryTree* left;
    struct BinaryTree* right;
}BTNode;
//读取字符串,并连接二叉树各个节点
BTNode* BinaryTreeCreat(char* arr,int* pi)
{
    if(arr[*pi] == '#')
    {
        (*pi)++;
        return NULL;
    }
    BTNode* node = (BTNode*)malloc(sizeof(BTNode));
    if(node == NULL)
    {
        perror("malloc()::fail");
        exit(-1);
    }
    node->val = arr[*pi];
    (*pi)++;
    node->left = BinaryTreeCreat(arr, pi);
    node->right = BinaryTreeCreat(arr, pi);
    return node;
}
//中序遍历打印
void BinaryTreeInOrder(BTNode* root)
{
    if(root == NULL)
        return;
    BinaryTreeInOrder(root->left);
    printf("%c ",root->val);
    BinaryTreeInOrder(root->right);
}

int main() {
    char arr[101];
    scanf("%s",arr);
    int pi = 0;
    BTNode* tree = BinaryTreeCreat(arr,&pi);
    BinaryTreeInOrder(tree);
    return 0;
}

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

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

相关文章

GoZero微服务个人探究之路(九)api文件编写总结

参考来源go-zero官方文档https://go-zero.dev/docs/tutorials 前言 go-zero是目前star最多的go语言微服务框架&#xff0c;api 是 go-zero特殊的语言&#xff0c;类型文件&#xff0c;go-zero自带的goctl可以通过.api文件生成http服务代码 api文件内容编写 不可使用关键字 …

webug存在的越权漏洞-水平越权以及垂直越权的漏洞复现(超详解)

越权漏洞-webug、 1.登录 账号&#xff1a;admin 密码&#xff1a;admin 2.进入逻辑漏洞 3.进入越权修改密码靶场 &#xff08;1&#xff09;输入账号密码 进入进去会发现没有权限进入 方法一&#xff1a; 这里我们只需要将 127.0.0.1:8080/control/a/auth_cross/cross_a…

小迪安全23WEB 攻防-Python 考点CTF 与 CMS-SSTI 模版注入PYC 反编译

#知识点&#xff1a; 1、PYC 文件反编译 2、Python-Web-SSTI 3、SSTI 模版注入利用分析 各语言的SSIT漏洞情况&#xff1a; SSIT漏洞过程&#xff1a; https://xz.aliyun.com/t/12181?page1&time__1311n4fxni0Qnr0%3DD%2FD0Dx2BmDkfDCDgmrYgBxYwD&alichlgrefhtt…

Angular响应式表单表单验证触发另一个字段校验

Angular响应式表单校验联动 前言表单字段日期校验函数效果 前言 在某些业务场景中&#xff0c;校验某表单字段的同时也需要校验另外一个与之相关的字段&#xff0c;例如开始时间和结束时间&#xff0c;要求结束时间必须晚于开始时间。在angular 响应式表单中改如何实现该需求呢…

云组态监控平台:开启智能监控新时代

在数字化浪潮中&#xff0c;物联网技术正逐渐成为各行业转型升级的核心驱动力。而云组态监控平台作为物联网技术的重要组成部分&#xff0c;正在开启智能监控的新时代。HiWoo Cloud的云组态监控平台&#xff0c;凭借其强大的功能和创新能力&#xff0c;致力于推动智能监控技术的…

uniapp app更新

uniapp app更新 这个版本要随之增加&#xff0c;不然刚更新时直接用app, 新包增加的那些页面跳转会有问题&#xff0c;不能跳新的页面 //app更新检测 updataApp(){const that this;uni.showLoading({title:加载中...})plus.runtime.getProperty(plus.runtime.appid, functio…

【微信小程序】图片违法违规内容鉴别(云函数)

微信小程序通过云调用校验一张图片是否含有违法违规内容。 选择图片: wx.chooseImage({count: 6,sizeType: [compressed], // 可以指定是原图还是压缩图&#xff0c;默认二者都有sourceType: [album, camera], // 可以指定来源是相册还是相机&#xff0c;默认二者都有success: …

用于医学分割的实时Test-time adaption

机构&#xff1a;约翰霍普金斯 论文&#xff1a;https://arxiv.org/abs/2203.05574 代码&#xff1a;https://github.com/jeya-maria-jose/On-The-Fly-Adaptation 摘要 基于深度学习的医学成像解决方案的一个主要问题是&#xff0c;当模型在不同于其训练的数据分布上进行测…

紫光展锐T760_芯片性能介绍_展锐T760安卓核心板定制

展锐T760核心板是一款基于国产5G芯片的智能模块&#xff0c;采用紫光展锐T760制程工艺为台积电6nm工艺&#xff0c;支持工艺具有出色的能效表现。其采用主流的44架构的八核设计&#xff0c;包括4颗2.2GHz A76核心和4颗A55核心设计&#xff0c;内存单元板载可达8GB Ram256GB ROM…

HCIA——29HTTP、万维网、HTML、PPP、ICMP;万维网的工作过程;HTTP 的特点HTTP 的报文结构的选择、解答

学习目标&#xff1a; 计算机网络 1.掌握计算机网络的基本概念、基本原理和基本方法。 2.掌握计算机网络的体系结构和典型网络协议&#xff0c;了解典型网络设备的组成和特点&#xff0c;理解典型网络设备的工作原理。 3.能够运用计算机网络的基本概念、基本原理和基本方法进行…

【python爬虫】爬虫编程技术的解密与实战

​&#x1f308;个人主页&#xff1a;Sarapines Programmer&#x1f525; 系列专栏&#xff1a; 爬虫】网络爬虫探秘⏰诗赋清音&#xff1a;云生高巅梦远游&#xff0c; 星光点缀碧海愁。 山川深邃情难晤&#xff0c; 剑气凌云志自修。 目录 &#x1f33c;实验目的 &#x1f…

Android开发修炼之路——(一)Android App开发基础-2

本专栏文章 上一篇 Android开发修炼之路——&#xff08;一&#xff09;Android App开发基础-1 2 App的工程结构 本节介绍App工程的基本结构及其常用配置&#xff0c;首先描述项目和模块的区别&#xff0c;以及工程内部各目录与配置文件的用途说明&#xff1b;其次阐述两种级别…

BabylonJS 6.0文档 Deep Dive 摄像机(六):遮罩层和多相机纹理

1. 使用遮罩层来处理多个摄影机和多网格物体 LayerMask是分配给每个网格&#xff08;Mesh&#xff09;和摄像机&#xff08;Camera&#xff09;的一个数。它用于位&#xff08;bit&#xff09;级别用来指示灯光和摄影机是否应照射或显示网格物体。默认值为0x0FFFFFFF&#xff…

SpringBoot使用druid

SpringBoot使用druid 一、前言二、配置1、pom依赖2、配置文件yml3、配置类 一、前言 Java程序很大一部分要操作数据库&#xff0c;为了提高性能操作数据库的时候&#xff0c;又不得不使用数据库连接池。 Druid 是阿里巴巴开源平台上一个数据库连接池实现&#xff0c;结合了 C…

【wink】草稿损坏如何恢复?

为节省储存空间&#xff0c;我们不会缓存您的原始视频。 原视频被删除或上传云盘后&#xff0c;可能会由于读取不到原视频而提示草稿损坏。 草稿损坏后&#xff0c;您可以尝试以下方法进行恢复&#xff1a; 从相册「最近删除」中恢复原视频&#xff1b;从云盘中下载原视频&…

单元测试——题目十二

目录 题目要求: 定义类 测试类 题目要求: 根据下列流程图编写程序实现相应处理,执行j=10*x-y返回文字“j1=:”和计算值,执行j=(x-y)*(10⁵%7)返回文字“j2=:”和计算值,执行j=y*log(x+10)返回文字“j3=:”和计算值。 编写程序代码,使用JUnit框架编写测试类对编写的…

力扣646. 最长数对链

动态规划 思路&#xff1a; 思路与 力扣354. 俄罗斯套娃信封问题 类似将序列进行排序&#xff0c;然后假设 dp[i] 为第 i 个元素的最长数对链个数&#xff1b;则其状态转移方程&#xff1a; 第 i 个元素之前的某一个元素&#xff08;假设是下标是 j&#xff09;&#xff0c;如…

残留扭矩测量方法有哪些——SunTorque智能扭矩系统

残留扭矩是指在设备或机器的转动部分停机后仍然存在的扭矩&#xff0c;通常是由于摩擦力、粘性阻力等因素引起的。残留扭矩测量是设备维护和故障诊断的重要环节&#xff0c;SunTorque智能扭矩系统一起和大家学习了解几种常见的残留扭矩测量方法。 suntoruqe智能扭矩系统 静态扭…

C++技术要点总结, 面试必备, 收藏起来慢慢看

目录 1. 语言对比 1.1 C 11 新特性 2.2 C 和 C 的区别 2.3 Python 和 C 的区别 2. 编译内存相关 2.1. C 程序编译过程 2.2. C 内存管理 2.3. 栈和堆的区别 2.4. 变量的区别 2.5. 全局变量定义在头文件中有什么问题&#xff1f; 2.6. 内存对齐 2.7. 什么是内存泄露 …

Tomcat运维

目录 一、Tomcat简介 二、系统环境说明 1、关闭防火墙&#xff0c;selinux 2、安装JDK 3、安装Tomcat 三、Tomcat目录介绍 1、tomcat主目录介绍 2、webapps目录介绍 3、Tomcat配置介绍&#xff08;conf&#xff09; 4、Tomcat的管理 四、Tomcat 配置管理页面(了解) …