【数据结构】二叉树的四种遍历方式——必做题

news2024/9/30 3:26:35

写在前面

学完上一篇文章的二叉树的遍历之后,来尝试下面的习题吧


开始做题

144. 二叉树的前序遍历 - 力扣(LeetCode)

94. 二叉树的中序遍历 - 力扣(LeetCode)

145. 二叉树的后序遍历 - 力扣(LeetCode)

为什么将这三道题放在一起呢?因为只要你学会了前序遍历,那么中序和后序遍历就自然不在话下(修改递归顺序),我就不写三个解析水字数了😅,一箭三雕,举一反三的时刻到了😁

敬告:先去做完,再看解析。为方便判错,力扣上的题加了很多限制,有些变量必要难以理解

给你二叉树的根节点 root ,返回它节点值的 前序 遍历。

题目简简单单就一句话,正是上一篇文章所讲的代码,但是参数却有些让人看不懂,我们通常这样写


void preorderTraversal(struct TreeNode* root) 
{
    if (root == NULL) return;  // 如果节点为空,返回
 
    printf("%d ", root->val);  // 先访问根节点
    preorderTraversal(root->left);  // 再递归遍历左子树
    preorderTraversal(root->right);  // 最后递归遍历右子树

但是力扣上的解析大都是这样的,为什么?

void PreOrder(struct TreeNode* root,int* arr,int* returnSize)
{
    if(root==NULL) 
       return;
    arr[(*returnSize)++]=root->val;
    PreOrder(root->left,arr,returnSize);
    PreOrder(root->right,arr,returnSize);
}

int* preorderTraversal(struct TreeNode* root, int* returnSize)
{
    int *arr=malloc(sizeof(int)*2000); //定义数组,用于存储遍历的返回值
    *returnSize=0;
    PreOrder(root,arr,returnSize);
    return arr;
}

原因:要求不同造成,需要更多参数;

  • 首先这道题不像上一篇文章我们写的那样,直接打印出来结点数值,而力扣要求将结点数值存进一个数组,返回时还得返回整个数组,这样我们就得先定义一个数组arr ,又因为需要malloc,故需要将数组定义成指针的形式;

  • 那returnSize的意思就是数组的长度,为什么要用指针呢?因为只有传值调用,指针接收,才能通过函数内returnSize的改变,进而影响returnSize被原本值,在这里初始值设为0就很巧妙,因为可以当成数组的下标;

  • 至于前序遍历在这里就可以必须要时刻更新这三个参数,干脆就直接再封装成一个函数来进行递归调用。

中序遍历和后序遍历就很简单了,一箭三雕,举一反三的时刻到了😁


102. 二叉树的层序遍历 - 力扣(LeetCode)

给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。

解析:这道题的难度有点大,耐心做

/**
 * Definition for a binary tree node.
 * struct TreeNode {
 *     int val;
 *     struct TreeNode *left;
 *     struct TreeNode *right;
 * };
 */

/**
 * Return an array of arrays of size *returnSize.
 * The sizes of the arrays are returned as *returnColumnSizes array.
 * Note: Both returned array and *columnSizes array must be malloced, assume caller calls free().
 */

//用队列的顺序存储结构来实现,本身问题很简单,但上面的三条英文注释让整个题变得不简单起来
/*

补充知识:二位动态整形数组的申请方法
int** arr=(int **)malloc(sizeof(int*)*3000);
arr[*returnSize] = (int*)malloc(sizeof(int) * (rear-front));
*/
int** levelOrder(struct TreeNode* root, int* returnSize, int** returnColumnSizes)
{
    *returnSize=0;  //数组大小设置为0
    if(root==NULL)  //根结点为空,没有遍历的必要,直接返回空
       return NULL;
    struct TreeNode *queue[3000];  //定义一个队列
    int **arr=(int**)malloc(sizeof(int*)*3000),front=0,rear=0;//定义一个二维数组
    *returnColumnSizes=(int*)malloc(sizeof(int)*3000);
    //根结点入队
    queue[rear++]=root;

    while(front!=rear)//这个是大循环
    {
        int temp=rear,k=0;
        arr[*returnSize] = (int*)malloc(sizeof(int) * (rear-front));
        while(front<temp)//rear会变化
        {
           struct TreeNode *p=queue[front++];//出队
           arr[*returnSize][k++] = p->val;
           if(p->left) //如果左结点不空,左节点入队
           {
             queue[rear++]=p->left;
           }
           if(p->right)  //如果右节点不空,右节点入队
           {
             queue[rear++]=p->right;
           }
        }
        (*returnColumnSizes)[*returnSize] = k;
        (*returnSize)++;
    }
    return arr;
}
  • 问:这段代码中returnSize和returnColumnSizes是什么意思?

  1. returnSize是一个指向整数的指针,用于存储二维数组arr的行数,即层序遍历结果的大小。在函数执行完毕后,通过修改returnSize来告知调用者结果的大小。

  1. returnColumnSizes是一个指向整数指针的指针,用于存储每一层中节点的数目。
    returnColumnSizes的指向的整数数组的大小应与arr的行数相同,即表示每一行中节点的数目。由于每一层的节点数目不同,因此我们需要动态分配一个整数数组来存储每一层中节点的数目。函数执行完毕后,returnColumnSizes指向的整数数组将包含每一层中节点的数目。

  1. 需要注意的是,如果函数中没有任何一层节点,那么returnSize将为0,而returnColumnSizes指向的数组应为NULL。

  • 问:为什么前者使用一级指针,后者使用二级指针?

  1. 在这段代码中,returnSize使用一级指针,而returnColumnSizes使用二级指针,是因为它们所表示的信息不同。

  1. returnSize指向的是整数类型的变量,用于表示层序遍历结果的行数,是一个单独的变量。因此,我们只需要通过一级指针将这个变量传递给函数,函数就可以修改这个变量的值,从而返回结果的行数。

  1. returnColumnSizes指向的是整数类型的指针数组,用于表示每一行中节点的数目,是一个数组。由于这个数组的大小在函数调用时未知,因此我们需要使用二级指针将它传递给函数。在函数内部,我们需要先分配一个整数类型的数组,然后将这个数组的地址存储到returnColumnSizes所指向的地址中。因此,我们需要使用二级指针。

  • int** arr=(int **)malloc(sizeof(int*)*3000);
    arr[*returnSize] = (int*)malloc(sizeof(int) * (rear-front));

  1. 第一行代码定义了一个名为arr的指向指针的指针变量,分配了一个大小为sizeof(int*)3000字节的内存,该内存用于存储二维数组的每个一维数组的地址

  1. 第二行代码则是为二维数组的一维数组分配内存,通过指针访问到第returnSize行,然后调用malloc函数分配一块大小为sizeof(int) * (rear-front)字节的内存,该内存用于存储一维数组中的元素,返回值为一维数组的首地址,使用强制类型转换将其转换为int类型,最后存储在arr[*returnSize]中,用于后续的访问。


写在最后

👍🏻 点赞,你的认可是我创作的动力!
收藏,你的青睐是我努力的方向!
✏️ 评论,你的意见是我进步的财富!

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

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

相关文章

Git使用笔记

分支branch切换到另一个分支git checkout 你要切换到的分支的名字git checkout master将本地的这个分支branch1和gitee上的branch1进行合并&#xff08;本地的branch1有的&#xff0c;gitee上branch1没有的增加上去&#xff09;git merge branch1git merge 分支的名字查看本地是…

图解LeetCode——剑指 Offer 25. 合并两个排序的链表

一、题目 输入两个递增排序的链表&#xff0c;合并这两个链表并使新链表中的节点仍然是递增排序的。 二、示例 2.1> 示例1&#xff1a; 【输入】1->2->4, 1->3->4 【输出】1->1->2->3->4->4 限制&#xff1a; 0 < 链表长度 < 1000 三、…

1、RocketMQ概述

文章目录1 MQ概述1.1 MQ简介1.2 MQ用途1.3 常见MQ产品1.4 MQ常见协议2 RocketMQ概述2.1 RocketMQ简介2.2 RocketMQ发展历程尚硅谷RocketMQ教程-讲师&#xff1a;Reythor雷&#xff08;老雷&#xff09; 我们缺乏的不是知识&#xff0c;而是学而不厌的态度 1 MQ概述 1.1 MQ简介…

实例9:四足机器人运动学正解平面RR单腿可视化

实例9&#xff1a;四足机器人正向运动学单腿可视化 实验目的 通过动手实践&#xff0c;搭建mini pupper四足机器人的腿部&#xff0c;掌握机器人单腿结构。通过理论学习&#xff0c;熟悉几何法、旋转矩阵法在运动学正解&#xff08;FK&#xff09;中的用处。通过编程实践&…

一个好的python文件可以有几种用途?

大家好鸭&#xff01;我是小熊猫~ 这次来带大家浅浅回顾一点python小知识~ 源码资料电子书:点击此处跳转文末名片获取 python文件总共有两种用途&#xff1a; 一种是执行文件另一种是被当做模块导入 编写好的一个python文件可以有两种用途&#xff1a; 1. 脚本&#xff0c;…

Laravel框架02:路由与控制器

Laravel框架02&#xff1a;路由与控制器一、路由配置文件二、路由参数三、路由别名四、路由群组五、控制器概述六、控制器路由七、接收用户输入一、路由配置文件 以web网页路由文件为例&#xff1a; 默认根路由 路由定义格式Route::请求方式(请求的URL, 匿名函数或控制响应的方…

【MongoDB】一、MongoDB的安装与部署

【MongoDB】一、MongoDB的安装与部署实验目的实验内容实验步骤一、下载MongoDB安装包二、创建文件夹data及子文件夹db和log三、启动MongDB服务1. 在命令行窗口执行启动MongoDB服务命令2. 打开mongodb.log3. 打开浏览器进行启动验证四、登录MongoDB五、配置环境变量六、将MongDB…

yii-shopwind商城多数页面报错,修改mysql一个配置就解决!

解决办法打开mysql配置文件&#xff0c;在[mysqld]下添加如下一行&#xff1a;sql_modeNO_ENGINE_SUBSTITUTION,STRICT_TRANS_TABLES成功解决!还以为是网站的代码问题, 惊讶到我了. 开源网站下载下来就报错 多不可思议. 终于是配置的问题!加油报错信息如下是其中一个界面的&…

银行数字化转型导师坚鹏:招商银行数字化转型战略研究

招商银行数字化转型战略研究课程背景&#xff1a; 很多银行存在以下问题&#xff1a; 不清楚如何制定银行数字化转型战略&#xff1f; 不知道其它银行的数字化转型战略是如何演变的&#xff1f; 课程特色&#xff1a; 用实战案例解读招商银行数字化转型战略。 用独特视角解…

【自然语言处理】基于句子嵌入的文本摘要算法实现

基于句子嵌入的文本摘要算法实现人们在理解了文本的含义后&#xff0c;很容易用自己的话对文本进行总结。但在数据过多、缺乏人力和时间的情况下&#xff0c;自动文本摘要则显得至关重要。一般使用自动文本摘要的原因包括&#xff1a; 减少阅读时间根据摘要&#xff0c;选择自…

Widget中的State-学习笔记

Widget 有 StatelessWidget 和 StatefulWidget 两种类型。StatefulWidget 应对有交互、需要动态变化视觉效果的场景&#xff0c;而 StatelessWidget 则用于处理静态的、无状态的视图展示。StatefulWidget 的场景已经完全覆盖了 StatelessWidget&#xff0c;因此我们在构建界面时…

如何把Windows自带壁纸、锁屏壁纸取出作为自己的桌面壁纸?(附:批量更改文件后缀名的方法)

这是一个目录0.0Windows桌面自带壁纸&#xff08;个性化中的壁纸&#xff09;Windows聚焦的锁屏壁纸批量更改后缀名的办法最近开机时&#xff0c;发现Windows聚焦的锁屏壁纸都很好看&#xff0c;特别想留下来当桌面壁纸&#xff0c;寻找了一下之后&#xff0c;终于找到了储存路…

第一节 变量、数据类型、关键字标识符

变量 什么是变量&#xff1a;就是用来存储一个数据的内存区域&#xff0c;而且里面存储的数据可以变化。 double money 10.0&#xff1b; 变量使用的注意事项&#xff1a; 1.先声明再使用。 2.变量声明后&#xff0c;就不能存储其他数据。 3.变量的有效范围是从定义开始到“}…

Dubbo源码解析-——SPI机制

文章目录一、什么是SPI机制二、Java原生的SPI机制2.1、javaSPI示例2.1.1、编写接口和实现类2.1.2、编写配置文件2.1.3、通过SPI机制加载实现类2.1.4、JAVA SPI 源码解析2.1.4.1、ServiceLoader#load2.1.4.2、ServiceLoader构造方法2.1.4.3、ServiceLoader#reload2.1.4.4、LazyI…

IoT项目系统架构案例2

项目背景 1.这个项目是对之前的案例的升级改造参考&#xff1a;IoT项目系统架构案例_iot案例_wxgnolux的博客-CSDN博客2.基于方案1的项目实施过程中碰到的问题,对硬件设备标准化的理念及新的功能需求(如根据天气预报温度调水温,APP界面可操作性优化等)•采用目前IoT主流厂商的架…

java的一些冷知识

接口并没有继承Object类首先接口是一种特殊的类&#xff0c;理由就是将其编译后是一个class文件大家都知道java类都继承自Object&#xff0c;但是接口其实是并没有继承Object类的 可以自己写代码测试: 获取接口类的class对象后遍历它的methods&#xff0c;可以发现是不存在Obje…

Java EE|TCP/IP协议栈之网络层IP协议详解

文章目录一、IP协议感性认知简介特点二、IP协议报头结构4位版本4位首部长度8位服务类型16位总长度16位标识8位生存时间&#xff08;TTL&#xff09;8位协议16位首部长度32位源ip&32位目的地址三、网络地址管理网段划分路由选择参考一、IP协议感性认知 简介 P是整个TCP/IP…

【云原生】k8s的pod基本概念

一、资源限制 Pod 是 kubernetes 中最小的资源管理组件&#xff0c;Pod 也是最小化运行容器化应用的资源对象。一个 Pod 代表着集群中运行的一个进程。kubernetes 中其他大多数组件都是围绕着 Pod 来进行支撑和扩展 Pod 功能的&#xff0c;例如用于管理 Pod 运行的 StatefulSe…

小说网站测试

目录 通用测试点 登录页面测试 接口测试 UI测试 注册页面 接口自动化 UI测试 忘记密码页面 接口测试 UI测试 修改密码页面 进行接口测试 UI测试 主页页面测试 分类页面测试 ​查询页面测试 作者页面测试 阅读小说页面测试 书架页面测试 通用测试点 登录页面测试 接…

k8s-资源限制-探针检查

文章目录一、资源限制1、资源限制的使用2、reuqest资源&#xff08;请求&#xff09;和limit资源&#xff08;约束&#xff09;3、Pod和容器的资源请求和限制4、官方文档示例5、资源限制实操5.1 编写yaml资源配置清单5.2 释放内存&#xff08;node节点&#xff0c;以node01为例…