《数据结构风云》:二叉树遍历的底层思维>递归与迭代的双重视角

《数据结构风云》:二叉树遍历的底层思维>递归与迭代的双重视角


在这里插入图片描述


🔥@晨非辰Tong: 个人主页
👀专栏:《C语言》《数据结构与算法入门指南》
💪学习阶段:C语言、数据结构与算法初学者
⏳“人理解迭代,神理解递归。”


文章目录

引言

二叉树遍历是理解树结构操作的基础,也是算法设计核心环节。前、中与后序遍历,以不同顺序访问节点,体现了递归与迭代思想的精髓。掌握转换规律,不仅能提升对数据结构本质的理解,更能为解决复杂算法问题奠定基础。本文将通过经典题目,解析如何还原二叉树,深入剖析遍历背后的逻辑与实现方法。
获取原码》点我《!!!

知识点前瞻

在这里插入图片描述

一、不一样的前序遍历

144. 二叉树的前序遍历

1.要求描述:

在这里插入图片描述

2.实现示例:

在这里插入图片描述

3.算法思路:

首先看平台给出的接口实现框架——>int* preorderTraversal(struct TreeNode*root, int* returnSize),这时候再看输出示例:返回的是一个数组,那么框架应该就是来返回数组的。对于returnSize猜测是目标树的节点个数,但是输出中没有给出,那么要自己去实现求个数接口
然后根据求出的节点个数去开辟数组空间(因为Note: The returned array must be malloced, assume caller calls free().)。
最后,就要实现前序遍历,但这个前序遍历与之前实现不太一样:不需要打印出节点的数值,只需要将数值存储在要返回的数组中。

复杂度:

  • 时间复杂度: O(N);
  • 空间复杂度: O(N);

3.1 具体代码实现

/** * 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(). *///求节点个数接口intBinaryTreeSize(structTreeNode* root){//根节点为空,树为空if(root ==NULL){return0;}//不为空,遍历左右子树//实质上就是根节点个数的累加return1+BinaryTreeSize(root->left)+BinaryTreeSize(root->right);}//前序遍历接口voidPreOrder(structTreeNode* root,int* arr,int* pi){//空树,直接返回if(root ==NULL){return;}//将非空节点的值存储在要返回的数组 arr[(*pi)++]= root->val;//遍历子树PreOrder(root->left, arr, pi);PreOrder(root->right, arr, pi);}//返回数组接口//returnSize:树的节点个数,输入为给出,代表要自己计算int*preorderTraversal(structTreeNode* root,int* returnSize){//调用函数求节点个数,开辟空间*returnSize =BinaryTreeSize(root);//开辟空间int* arr =(int*)malloc(sizeof(int)*(*returnSize));//前序遍历int i =0;PreOrder(root, arr,&i);return arr;}
在这里插入图片描述

3.2 注意要点

  1. 变量i的定义、传参:程序中数据存放在数组中需要下标i,并没有在前序遍历接口中创建或者创建全局变量,而是通过传参(会导致i重复初始化或者累加,前面说过)。但是传的是地址,如果只传数值的话,在后续递归调用函数,这个i不会随着元素的增加改变.

二、不一样的中序遍历

94. 二叉树的中序遍历

1.要求描述:

在这里插入图片描述

2.实现示例

在这里插入图片描述

3.算法思路:

整体思路与上面的前序遍历大致相同,只需要将 arr[(*pi)++] = root->val;放在PreOrder(root->left, arr, pi); PreOrder(root->right, arr, pi);中间即可。实现左根右。
复杂度:

  • 时间复杂度: O(N);
  • 空间复杂度: O(N);

3.1 具体代码实现:

/** * 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(). *///求节点个数接口intBinaryTreeSize(structTreeNode* root){//根节点为空,树为空if(root ==NULL){return0;}//不为空,遍历左右子树//实质上就是根节点个数的累加return1+BinaryTreeSize(root->left)+BinaryTreeSize(root->right);}//中序遍历voidInOrder(structTreeNode* root,int* arr,int* pi){//树为空if(root ==NULL){return;}//树不为空//遍历左子树InOrder(root->left, arr, pi); arr[(*pi)++]= root->val;InOrder(root->right, arr, pi);}int*inorderTraversal(structTreeNode* root,int* returnSize){//调用函数求节点个数,开辟空间*returnSize =BinaryTreeSize(root);//开辟空间int* arr =(int*)malloc(sizeof(int)*(*returnSize));//中序遍历int i =0;InOrder(root, arr,&i);return arr;}

三、不一样的后序遍历

145. 二叉树的后序遍历

1.要求描述:

在这里插入图片描述

2.实现示例:

在这里插入图片描述

3.算法思路:

整体思路与上面的中序遍历大致相同,只需要将 arr[(*pi)++] = root->val;放在PreOrder(root->left, arr, pi); PreOrder(root->right, arr, pi);后面即可。实现左右根。
复杂度:

  • 时间复杂度: O(N);
  • 空间复杂度: O(N);

3.1 具体代码实现:

/** * 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(). *///求节点个数接口intBinaryTreeSize(structTreeNode* root){//根节点为空,树为空if(root ==NULL){return0;}//不为空,遍历左右子树//实质上就是根节点个数的累加return1+BinaryTreeSize(root->left)+BinaryTreeSize(root->right);}//后序遍历voidPostOrder(structTreeNode* root,int* arr,int* pi){//树为空if(root ==NULL){return;}//树不为空//遍历左子树PostOrder(root->left, arr, pi);PostOrder(root->right, arr, pi); arr[(*pi)++]= root->val;}int*postorderTraversal(structTreeNode* root,int* returnSize){//调用函数求节点个数,开辟空间*returnSize =BinaryTreeSize(root);//开辟空间int* arr =(int*)malloc(sizeof(int)*(*returnSize));//后序遍历int i =0;PostOrder(root, arr,&i);return arr;}
在这里插入图片描述

四、二叉树遍历

TSINGK110 二叉树遍历

1.要求描述:

在这里插入图片描述

2.实现示例:

在这里插入图片描述

3.算法思路:

–牛客网平台全部代码都需要我们自己去实现,比较麻烦。 首先,根据描述:我们需要将用户输入的前序遍历完成的字符串存放在数组中,再根据数组来重现树的结构——>自定义创建树函数。创建树就需要知道树节点的结构,再申请节点——>定义树节点的结构、自定义创建节点函数。
上面这些函数的实现,我们前面都操作过。 然后,就是要中序遍历,这个我们也实现过。

复杂度:

  • 时间复杂度:O(N) ;
  • 空间复杂度:O(N);

3.1具体代码实现

#include<stdio.h>#include<stdlib.h>//定义二叉树的结构typedefstructBinaryTreeNode{char data;structBinaryTreeNode* left;structBinaryTreeNode* right;}BTNode;//根据字符创建节点 BTNode*buyNode(char ch){ BTNode* newnode =(BTNode*)malloc(sizeof(BTNode)); newnode->data = ch; newnode->left = newnode->right =NULL;return newnode;}//创建树 BTNode*creatTree(char* arr,int* pi){//如果是空节点,返回空if(arr[*pi]=='#'){(*pi)++;returnNULL;}//创建新节点--根左右 BTNode* root =buyNode(arr[(*pi)++]); root->left =creatTree(arr, pi); root->right =creatTree(arr, pi);return root;//最终返回指向根节点的指针}//中序遍历voidInOrder(BTNode* root){if(root ==NULL){return;}//左右根InOrder(root->left);printf("%c ", root->data);InOrder(root->right);}intmain(){//读取用户输入的字符串char arr[100];scanf("%s", arr);//根据字符串数组创建二叉树(前序遍历的)int i =0;//接收创建树的根节点 BTNode* root =creatTree(arr,&i);//中序遍历InOrder(root);return0;}

总结

🍓 我是晨非辰Tong!若这篇技术干货帮你打通了学习中的卡点: 👀 【关注】跟我一起深耕技术领域,从基础到进阶,见证每一次成长 ❤️ 【点赞】让优质内容被更多人看见,让知识传递更有力量 ⭐ 【收藏】把核心知识点、实战技巧存好,需要时直接查、随时用 💬 【评论】分享你的经验或疑问(比如曾踩过的技术坑?),一起交流避坑 🗳️ 【投票】用你的选择助力社区内容方向,告诉大家哪个技术点最该重点拆解 技术之路难免有困惑,但同行的人会让前进更有方向~愿我们都能在自己专注的领域里,一步步靠近心中的技术目标! 

二叉树遍历序列的互推,核心在于把握“后序定根,中序分左右”的规律。掌握这一原理,不仅能解决序列重建问题,更能深化对递归和树结构的理解,为学习更复杂的数据结构奠定基础。

Read more

Redis Hash 全解析:从入门到精通,解锁高性能对象存储的钥匙

Redis Hash 全解析:从入门到精通,解锁高性能对象存储的钥匙

前言 在现代应用开发中,我们无时无刻不在与“对象”打交道——用户信息、商品详情、配置项、会话数据……如何高效、清晰地在缓存或数据库中存储这些结构化数据,是每一个开发者都需要面对的课题。 你可能会想到将一个对象序列化成 JSON 字符串,然后用一个简单的 Key-Value 方式存入 Redis。这确实是一种方法,但当你只想修改对象的某一个属性(比如用户的积分)时,就不得不读取整个 JSON 字符串,反序列化成对象,修改属性,再序列化回 JSON,最后整个写回 Redis。这个过程不仅繁琐,而且在并发环境下极易引发数据覆盖问题,性能开销也相当可观。 那么,有没有一种更原生、更高效的方式来处理这种“对象”存储呢?答案是肯定的。Redis 为我们提供了一种强大的数据结构,它天生就是为了解决这类问题而生——Hash(哈希/散列)。 本文将带领你深入探索 Redis Hash

By Ne0inhk
数据结构【AVL树】

数据结构【AVL树】

AVL树 * 1.AVL树 * 1.1AVL的概念 * 1.2平衡因子 * 2.AVl树的实现 * 2.1AVL树的结构 * 2.2AVL树的插入 * 2.3 旋转 * 2.3.1 旋转的原则 1.AVL树 1.1AVL的概念 AVL树可以是一个空树。 它的左右子树都是AVL树,且左右子树的高度差的绝对值不超过1。AVL树是一颗高度平衡搜索二叉树,通过控制高度差去控制平衡。 AVL树整体结点数量和分布和完全二叉树类似,高度可以控制在 logN ,那么增删查改的效率也可以控制在O(logN ) ,相比二叉搜索树有了本质的提升。 1.2平衡因子 结点的平衡因子=右子树的高度减去左子树的高度,也就是说任何结点的平衡因子等于0/1/-1,AVL树并不是必须要平衡因子,但是有了平衡因子可以更方便我们去进行观察和控制树是否平衡,就像⼀个风向标一样。 为什么AVL树是高度平衡搜索二叉树,要求高度差不超过1,而不是高度差是0呢? 因为例如二和四个结点无法达到0.

By Ne0inhk
玩转二叉树:数据结构中的经典之作

玩转二叉树:数据结构中的经典之作

、 嘿嘿,家人们,今天咱们来详细剖析数据结构中的二叉树,好啦,废话不多讲,开干! 1:树的概念以及结构 1.1:树的概念 1.2:树的相关概念 1.3:树的表示 1.3.1:左孩子右兄弟表示法 2:二叉树的概念以及结构 2.1:概念 2.2:特殊的二叉树 2.3:二叉树的性质 2.4:二叉树的存储结构 2.4.1:顺序存储 2.4.2:链式存储 3:二叉树的顺序结构以及实现 3.1:二叉树的顺序结构 3.2:

By Ne0inhk