数据结构——链式二叉树(2)

news2024/11/18 0:13:38

目录

🍁一、二叉树的销毁

🍁二、在二叉树中查找某个数,并返回该结点

🍁三、LeetCode——检查两棵二叉树是否相等

🌕(一)、题目链接:100. 相同的树 - 力扣(LeetCode)

🌕(二)、解答:

🍁四、LeetCode——二叉树的前序遍历(与上一篇文章不太一样)

🌕(一)、题目链接:144. 二叉树的前序遍历 - 力扣(LeetCode)

🌕(二)、解答:


接上篇文章,我们接着学习关于链式二叉树的几种操作。

🍁一、二叉树的销毁

//销毁
void FreeDestroy(BTNode* root)
{
	if (root == NULL)
	{
		return;
	}
	FreeDestroy(root->left);
	FreeDestroy(root->right);
	free(root);
}

对于销毁,使用前序或者后序遍历都可以,但前序需要在销毁根结点的之前用临时指针保存根节点的左右子树,这样比较麻烦,所以最合适的还是后序,先销毁左右子树,然后才销毁根节点,这样按顺序的来就可以了。而我们使用递归最重要的是如何转换为子问题以及最小子递归的返回条件问题,这里很显然我们访问到NULL时就可以返回了,然后在依次销毁。

🍁二、在二叉树中查找某个数,并返回该结点

//在二叉树中找某个数的结点
BTNode* TreeFind(BTNode* root,int x)
{
	if (root == NULL)
		return NULL;
	if (root->val == x)
		return root;
	BTNode* tmp = TreeFind(root->left, x);
	if (tmp)
		return tmp;
	tmp = TreeFind(root->right, x);
	if (tmp)
		return tmp;
	return NULL;
}

①:依然是遍历的思路,这里我们选择前序遍历,先将根节点比较了再去左右子树比较;

②:然后最小子递归返回条件也是当root为空时,返回NULL,表示该条路径中没有找到数x;

③:若当前结点的数据等于x,表示找到了,就返回该结点;

④:若没找到,就先去左子树找,若左子树返回值不为NULL,说明找到了,返回左孩子结点;若左子树没找到,则返回NULL,不进如if语句;

⑤:左子树没找到,就开始找右子树,和左子树同样的道理,找到返回右子树结点,没找到返回NULL;

🍁三、LeetCode——检查两棵二叉树是否相等

🌕(一)、题目链接:100. 相同的树 - 力扣(LeetCode)

🌕(二)、解答:

bool isSameTree(struct TreeNode* p, struct TreeNode* q) {
    //两个树都为NULL
    if(p==NULL&&q==NULL)
    return true;
    //其中一个为NULL
    if(p==NULL||q==NULL)
    return false;
    //都不为NULL,则比较数据
    if(p->val!=q->val)
    //到这里说明该结点数据相同,则比较左子树,然后比较右子树
    return false;
    return isSameTree(p->left,q->left)&&
            isSameTree(p->right,q->right);
}

①:若两棵树都为NULL,则代表此结点的数据相同,返回ture;

②:若其中一个为NULL,另一个不为NULL,则两结点数据不同,返回false;

③:若两个都不为NULL,则比较数据,若数据不相同,则两结点数据不同,返回false;

④:若不为上述三种情况,则说明该结点相同,则比较其左子树,然后比较其右子树;

🍁四、LeetCode——二叉树的前序遍历(与上一篇文章不太一样)

🌕(一)、题目链接:144. 二叉树的前序遍历 - 力扣(LeetCode)

🌕(二)、解答:

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */
/**
 * Note: The returned array must be malloced, assume caller calls free().
 */

 //手动写的计算树的结点个数函数
 int TreeSize(struct TreeNode*root)
 {
     return root==NULL?0:TreeSize(root->left)+TreeSize(root->right)+1;
 }

 //手动写的遍历函数
void preorder(struct TreeNode* root,int* a,int* pi)
{
    if(root==NULL)
    return;
    a[(*pi)++]=root->val;
    preorder(root->left,a,pi);
    preorder(root->right,a,pi);
}

//LeetCode给定函数
int* preorderTraversal(struct TreeNode* root, int* returnSize) {
    int n=TreeSize(root);
    int* a=(int*)malloc(sizeof(int)*n);

    int j=0;
    preorder(root,a,&j);

    *returnSize=n;
    return a;
}

二叉树的前序遍历,我们在上一篇文章已经讲解了,是一个很简单的过程;但这道题不一样,不一样在这里存在一些OJ题的技巧解释;

①:这道题的意思是叫我们把树的前序遍历的结果,存放在一个数组中,最后在返回该数组的起始地址;

②:我们看到LeetCode给定的函数接口中有两个形参,一个是待操作的树,另一个是整形指针,但我们看整形指针字面意思,应该知道我们应该返回一个数组的个数,那为什么会是一个整形指针呢?因为在接口外部传的是该变量地址,我们要在函数接口中根据待测树的结点数,去解引用整形指针改变函数外部的数组个数,所以使用了传址调用。

③:所以我们手动写了一个函数,计算出待测树的结点个数,然后解引用returnSize,将树的结点数赋值给它。

④:又因为我们要用到递归的思路,所以又手动写了一个函数,进行前序遍历及其将数据存到数组中的操作,又因为数组需要用到下标,为了下标j能在任意递归栈帧中改变,所以我们使用了传址调用,将下标j的地址传进去,这样,我们每赋值一次,就解引用后加1,这样下标就能动态改变;

⑤:接着便是前序遍历的算法,只是将打印换成了赋值给数组这一操作;

本次知识到此结束,希望对你有所帮助!

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

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

相关文章

MySQL十部曲之四:MySQL中的数据类型

文章目录 前言概述数字类型数字类型语法数字类型字面量十六进制字面量位字面量布尔字面量 数字类型的属性超出范围和溢出处理 时间和日期类型时间和日期类型语法DATE、DATETIME和TIMESTAMP的异同TIMESTAMP和DATETIME的自动初始化和更新时间和日期字面量 字符串类型字符串类型语…

Android 基础技术——Handler

笔者希望做一个系列,整理 Android 基础技术,本章是关于 Handler 为什么一个线程对应一个Looper? 核心:通过ThreadLocal保证 Looper.prepare的时候,ThreadLocal.get如果不空报异常;否则调用ThreadLocal.set,…

376. 摆动序列 - 力扣(LeetCode)

题目描述 如果连续数字之间的差严格地在正数和负数之间交替,则数字序列称为摆动序列。第一个差(如果存在的话)可能是正数或负数。少于两个元素的序列也是摆动序列。 例如, [1,7,4,9,2,5] 是一个摆动序列,因为差值 (6,…

项目中从需求分析到研发上线

一、背景 应用系统从设想到需求到研发到上线会经历一些列工程化过程。比如经典的瀑布模型工作流,其实就是一个经过很多经验总结下来的工程方法。本节阐述项目中从需求到研发上线的过程。但是也有些根据不同的行业,不同的公司,不同管理者的风…

Cesium加载地图-高德影像

废话不多说&#xff0c;直接上代码 整体代码 <template><div id"cesiumContainer" style"height: 100vh;"></div><div id"toolbar" style"position: fixed;top:20px;left:220px;"><el-breadcrumb><…

【数据结构与算法】5.详解双向链表的基本操作(Java语言实现)

&#x1f4da;博客主页&#xff1a;爱敲代码的小杨. ✨专栏&#xff1a;《Java SE语法》 ❤️感谢大家点赞&#x1f44d;&#x1f3fb;收藏⭐评论✍&#x1f3fb;&#xff0c;您的三连就是我持续更新的动力❤️ &#x1f64f;小杨水平有限&#xff0c;欢迎各位大佬指点&…

Oracle扩展ASM存储

物理增加1T存储到服务器。 , 绑定裸设备 vi /etc/udev/rules.d/99-asmdevice.rules KERNEL"dm-*",ENV{DM_UUID}"mpath-360002ac000000000000001a700006bc6",NAME"asm_arch",OWNER"grid",GROUP"asmadmin",MODE"0660&q…

第二百八十八回

文章目录 1. 概念介绍2. 使用方法2.1 实现步骤2.2 具体细节 3. 示例代码4. 内容总结 我们在上一章回中介绍了"如何获取文件类型"相关的内容&#xff0c;本章回中将介绍如何播放视频.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 播放视频是我们常用…

区块空间----流动性铭文

铭文正在改变加密世界&#xff0c;越来越多的人开始关注铭文&#xff0c;并参与进来&#xff01;铭文赛道的未来是非常具有潜力和想象力的&#xff0c;甚至能够达到加密货币的普及水平。当然&#xff0c;这需要更多的基础设施更多的用例支持&#xff0c;但是一切都才刚刚开始。…

Go语言grpc服务开发——Protocol Buffer

文章目录 一、Protocol Buffer简介二、Protocol Buffer编译器安装三、proto3语言指南四、序列化与反序列化五、引入grpc-gateway1、插件安装2、定义proto文件3、生成go文件4、实现Service服务5、gRPC服务启动方法6、gateway服务启动方法7、main函数启动8、验证 相关参考链接&am…

Mysql-日志介绍 日志配置

环境部署 docker run -d -p 3306:3306 --privilegedtrue -v $(pwd)/logs:/var/lib/logs -v $(pwd)/conf:/etc/mysql/conf.d -v $(pwd)/data:/var/lib/mysql -e MYSQL_ROOT_PASSWORD654321 --name mysql mysql:5.7运行指令的目录下新建好这些文件&#xff1a; 日志类型 日…

dhcp服务器的ip池的待分配ip地址是否冲突的检测机制

看到有的资料说&#xff0c;dhcp服务器在分配ip地址时&#xff0c;要检测是否待分配的ip地址是否存在冲突&#xff0c;会向广播域发出&#xff0c;对应ip发出icmp的ping消息来验证是否冲突。特地用自己的公司的交换机验证一下&#xff0c;在交换机上镜像抓包观察一下。 wiresha…

社交媒体与新闻:Facebook在信息传播中的角色

社交媒体的崛起不仅改变了人们的日常交流方式&#xff0c;也对新闻传播产生了深远的影响。在众多社交媒体平台中&#xff0c;Facebook以其庞大的用户基础和强大的社交网络机制&#xff0c;成为信息传播的中流砥柱。本文将深入探讨Facebook在社交媒体与新闻传播的交汇点上扮演的…

Qt应用开发(安卓篇)——调用java代码,使用安卓api

一、前言 在Qt on Android开发的时候&#xff0c;它不像在嵌入式linux&#xff0c;几乎全部的操作都是可以通过文件来完成。很多场景下的功能都需要使用安卓的API去实现&#xff0c;开发人员需要通过这些API进行编程&#xff0c;无需访问源码&#xff0c;或者理解内部的机制。比…

IP报文格式

IP报文格式 报文格式 图1 IP头格式 表1 IP头字段解释 字段长度含义Version4比特 4&#xff1a;表示为IPV4&#xff1b;6&#xff1a;表示为IPV6。IHL4比特首部长度&#xff0c;如果不带Option字段&#xff0c;则为20&#xff0c;最长为60&#xff0c;该值限制了记录路由选项。…

阅读go语言工具源码系列之gopacket(谷歌出品)----第二集 layers-巧妙的抽象与无聊的协议包

上一集中我们讲到了wpcap.dll的go封装方法&#xff0c;对于linux系统下libpcap的go封装采用的是常用的cgo方式&#xff0c;想了解的可以看看pcap文件夹中的pcap_unix.go。 我们得到了wpcap.dll的go调用&#xff0c;就可以利用它来进行列举所有网络设备&#xff0c;例如以下代码…

docker容器生命周期管理命令

文章目录 前言1、docker create2、docker run2.1、常用选项2.2、系统2.3、网络2.4、健康检查 3、docker start/stop/restart4、docker kill5、docker rm6、docker pause/unpause总结 前言 在云原生时代&#xff0c;Docker已成为必不可少的容器管理工具。通过掌握Docker常用的容…

派网AX50C做多宽带路由和核心交换机配置实战教程

接近300办公人员的工厂需要网络升级&#xff0c;我规划设计和部署实施了以下方案&#xff0c;同样是简约不简单&#xff0c;在满足性能需求稳定性的前提下&#xff0c;既有经济性&#xff0c;又有安全性。 派网做路由器&#xff0c;刚好开启默认防病毒策略&#xff0c;省下来一…

【QT+QGIS跨平台编译】之十一:【libzip+Qt跨平台编译】(一套代码、一套框架,跨平台编译)

文章目录 一、libzip介绍二、文件下载三、文件分析四、pro文件五、编译实践一、libzip介绍 libzip是一个开源C库,用于读取,创建和修改zip文件。 libzip可以从数据缓冲区,文件或直接从其他zip归档文件直接复制的压缩数据中添加文件。在不关闭存档的情况下所做的更改可以还原…

TCP 状态转换以及半关闭

TCP 状态转换&#xff1a; 上图中还没有进行握手的时候状态是关闭的。 三次握手状态的改变&#xff1a; 客户端发起握手。 调用 connect() 函数时状态转化为&#xff1a;SYN_SENT。调用 listen() 函数时状态转换为&#xff1a;LISTEN。ESTABLISHED是被连接的状态。 四次挥手…