牛客题解 | 二叉树的后序遍历

news/2025/2/22 14:57:25/文章来源:https://www.cnblogs.com/wc529065/p/18727226

题目

题目链接

题目的主要信息:
  • 给定一颗二叉树的根节点,输出其后序遍历的结果
举一反三:

学习完本题的思路你可以解决如下题目:

BM23.二叉树的前序遍历

BM24.二叉树的中序遍历

方法一:递归(推荐使用)

知识点:二叉树递归

递归是一个过程或函数在其定义或说明中有直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解。因此递归过程,最重要的就是查看能不能讲原本的问题分解为更小的子问题,这是使用递归的关键。

而二叉树的递归,则是将某个节点的左子树、右子树看成一颗完整的树,那么对于子树的访问或者操作就是对于原树的访问或者操作的子问题,因此可以自我调用函数不断进入子树。

思路:

什么是二叉树的后续遍历,简单来说就是“左右根”,展开来说就是优先访问根节点的左子树的全部节点,然后再访问根节点的右子树的全部节点,最后再访问根节点。对于每棵子树的访问也按照这个逻辑,因此叫做“左右根”的顺序。

从上述后序遍历的解释中我们不难发现,它存在递归的子问题:对每个子树的访问,可以看成对于上一级树的子问题。那我们可以用递归处理:

  • 终止条件: 当子问题到达叶子节点后,后一个不管左右都是空,因此遇到空节点就返回。
  • 返回值: 每次处理完子问题后,就是将子问题访问过的元素返回,依次存入了数组中。
  • 本级任务: 对于每个子问题,优先进入左子树的子问题,访问完了再进入右子树的子问题,最后回到父问题访问根节点。

具体做法:

  • step 1:准备数组用来记录遍历到的节点值,Java可以用List,C++可以直接用vector。
  • step 2:从根节点开始进入递归,遇到空节点就返回,否则优先进入左子树进行递归访问。
  • step 3:左子树访问完毕再进入根节点的右子树递归访问。
  • step 4:最后回到根节点,访问该节点。

Java实现代码:

import java.util.*;
public class Solution {public void postorder(List<Integer> list, TreeNode root){//遇到空节点则返回if(root == null) return;//先去左子树postorder(list, root.left); //再去右子树postorder(list, root.right); //最后访问根节点list.add(root.val); }public int[] postorderTraversal (TreeNode root) {//添加遍历结果的数组List<Integer> list = new ArrayList(); //递归后序遍历postorder(list, root); //返回的结果int[] res = new int[list.size()]; for(int i = 0; i < list.size(); i++)res[i] = list.get(i);return res;}
}

C++实现代码:

class Solution {
public:void postorder(vector<int> &res, TreeNode* root){//遇到空节点则返回if(root == NULL) return;//先遍历左子树postorder(res, root->left); //再遍历右子树postorder(res, root->right); //最后遍历根节点res.push_back(root->val); }vector<int> postorderTraversal(TreeNode* root) {vector<int> res;//递归后序遍历postorder(res, root);  return res;}
};

Python实现代码

class Solution:def postorder(self, list: List[int], root: TreeNode):# 遇到空节点则返回if not root:return# 先去左子树self.postorder(list, root.left) # 再去右子树self.postorder(list, root.right) # 最后访问根节点list.append(root.val) def postorderTraversal(self , root: TreeNode) -> List[int]:res = []# 递归后序遍历self.postorder(res, root) return res

复杂度分析:

  • 时间复杂度:\(O(n)\),其中\(n\)为二叉树的节点数,遍历二叉树所有节点
  • 空间复杂度:\(O(n)\),最坏情况下二叉树化为链表,递归栈深度为\(n\)
方法二:非递归(扩展思路)

知识点:栈

栈是一种仅支持在表尾进行插入和删除操作的线性表,这一端被称为栈顶,另一端被称为栈底。元素入栈指的是把新元素放到栈顶元素的上面,使之成为新的栈顶元素;元素出栈指的是从一个栈删除元素又称作出栈或退栈,它是把栈顶元素删除掉,使其相邻的元素成为新的栈顶元素。

思路:

既然二叉树的前序遍历和中序遍历都可以使用栈来代替递归,那后序遍历是否也可以呢?答案是可以的,但是会比前二者复杂一点点。

根据后序遍历“左右中”的顺序,那么后序遍历也与中序遍历类似,要先找到每棵子树的最左端节点:

//每次找到最左节点
while(root != NULL){ s.push(root);root = root->left;
}

然后我们就要访问该节点了嘛?不不不,如果它还有一个右节点呢?根据“左右根”的原则,我还要先访问右子树。我们只能说它是最左端的节点,它左边为空,但是右边不一定,因此这个节点必须被看成是这棵最小的子树的根。要怎么访问根节点呢?

我们都知道从栈中弹出根节点,一定是左节点已经被访问过了,因为左节点是子问题,访问完了才回到父问题,那么我们还必须要确保右边也已经被访问过了。如果右边为空,那肯定不用去了,如果右边不为空,那我们肯定优先进入右边,此时再将根节点加入栈中,等待右边的子树结束。

//该节点再次入栈
s.push(node);
//先访问右边
root = node->right;

不过,当右边被访问了,又回到了根,我们的根怎么知道右边被访问了呢?用一个前序指针pre标记一下,每个根节点只对它的右节点需要标记,而每个右节点自己本身就是一个根节点,因此每次访问根节点的时候,我们可以用pre标记为该节点,回到上一个根节点时,检查一下,如果pre确实是它的右子节点,哦那正好,刚刚已经访问过了,我现在可以安心访问这个根了。

//如果该元素的右边没有或是已经访问过
if(node->right == NULL || node->right == pre){ //访问中间的节点res.push_back(node->val); //且记录为访问过了pre = node; 
}

具体做法:

  • step 1:开辟一个辅助栈,用于记录要访问的子节点,开辟一个前序指针pre。
  • step 2:从根节点开始,每次优先进入每棵的子树的最左边一个节点,我们将其不断加入栈中,用来保存父问题。
  • step 3:弹出一个栈元素,看成该子树的根,判断这个根的右边有没有节点或是有没有被访问过,如果没有右节点或是被访问过了,可以访问这个根,并将前序节点标记为这个根。
  • step 4:如果没有被访问,那这个根必须入栈,进入右子树继续访问,只有右子树结束了回到这里才能继续访问根。

图示:

alt

Java实现代码:

import java.util.*;
public class Solution {public int[] postorderTraversal (TreeNode root) {//添加遍历结果的数组List<Integer> list = new ArrayList(); Stack<TreeNode> s = new Stack<TreeNode>();TreeNode pre = null;while(root != null || !s.isEmpty()){ //每次先找到最左边的节点while(root != null){ s.push(root);root = root.left;}//弹出栈顶TreeNode node = s.pop(); //如果该元素的右边没有或是已经访问过if(node.right == null || node.right == pre){ //访问中间的节点list.add(node.val); //且记录为访问过了pre = node; }else{//该节点入栈s.push(node); //先访问右边root = node.right; }}//返回的结果int[] res = new int[list.size()]; for(int i = 0; i < list.size(); i++)res[i] = list.get(i);return res;}
}

C++实现代码:

class Solution {
public:vector<int> postorderTraversal(TreeNode* root) {vector<int> res;//辅助栈stack<TreeNode*> s; TreeNode* pre = NULL; while(root != NULL || !s.empty()){ //每次先找到最左边的节点while(root != NULL){ s.push(root);root = root->left;}//弹出栈顶TreeNode* node = s.top(); s.pop();//如果该元素的右边没有或是已经访问过if(node->right == NULL || node->right == pre){ //访问中间的节点res.push_back(node->val); //且记录为访问过了pre = node; }else{//该节点入栈s.push(node);//先访问右边root = node->right; }}return res;}
};

Python实现代码

class Solution:def postorderTraversal(self , root: TreeNode) -> List[int]:# 添加遍历结果的数组res, s = [], [] pre = Nonewhile root or s:# 每次先找到最左边的节点while root: s.append(root)root = root.left# 弹出栈顶node = s[-1] s.pop()# 如果该元素的右边没有或是已经访问过if not node.right or node.right is pre: # 访问中间的节点res.append(node.val) # 且记录为访问过了pre = node else:# 该节点入栈s.append(node) # 先访问右边root = node.right return res

复杂度分析:

  • 时间复杂度:\(O(n)\),其中\(n\)为二叉树的节点数,遍历二叉树所有节点
  • 空间复杂度:\(O(n)\),辅助栈空间最大为链表所有节点数

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

FB内部使用地址寄存器案例 - 32位指针区域内间接寻址

案例内容:实现DQ32模块通道依次循环输出。利用循环左移指令,把默认值位16#1的DWORD循环左移,最后绑定QD,实现DQ32模块通道依次循环输出。限制1:不使用SCL实现,而用STL或LAD来实现。 限制2:DWORD被拆分为bit的时候会因大端模式导致的输出对应异常,需要重新梳理bit位和实际…

牛客题解 | 二分查找-I

牛客输入输出题单题解题目 题目链接 题目的主要信息:给定一个元素升序的、无重复数字的整型数组 nums 和一个目标值 target 找到目标值的下标 如果找不到返回-1举一反三: 学习完本题的思路你可以解决如下题目: BM18.二维数组中的查找 BM19.寻找峰值 BM21.旋转数组 方法:二分…

ML树构建简明教程

数据准备 Teamviewer登录实验室服务器,访问http://172.17.128.86:8501/CleanData,按照页面对应的格式要求分别从NCBI和GISAID数据库下载数据,拖拽到对应的位置,点击GO即可。Gisaid DNA Accession no.|DNA INSDC|Isolate name|Collection date|SegmentNCBI Format:>{acc…

windows11安装

准备工作 1.一个8G以上的U盘 2.可以上网的电脑 3.要安装的电脑接好可上网的网线开始:把U盘的资料先备份,因为制作U盘安装系统需要格式化U盘 打开微软官网下载 https://www.microsoft.com/zh-cn/software-download/windows11选择创建 Windows 11 安装媒体,会下载一个"me…

DeepSeek崛起:程序员“饭碗”被抢,还是职业进化新起点?

2025年伊始,Meta创始人扎克伯格的一则声明引发全球程序员热议:“AI将在今年达到中级工程师水平,逐步接管编程工作。”与此同时,国产AI大模型DeepSeek的爆火,让一名8岁女孩仅用45分钟开发出聊天机器人的案例刷屏全网。AI的代码能力已从“辅助工具”跃升为“协同开发者”,程…

java-调用火山引擎官方API

java-调用火山引擎官方API(字节跳动旗下的AI服务平台),AI模型包括:DeepSeek、豆包、其他。。。 2025-02-20 16:33:06 星期四一、官方平台入口: 入口如下:https://console.volcengine.com/user/basics/ 1>.使用方式:1.在开放平台注册账号具体按照指引及进行注册即可,包…

reactnative 手写签名报错

有没有大佬知道这种问题怎么解决!!!!! 在线等!!!

Ollama模型迁移

为了方便本地大模型部署和迁移,本文提供了一个关于Ollama的模型本地迁移的方法。由于直接从Ollama Hub下载下来的模型,或者是比较大的GGUF模型文件,往往会被切分成多个,而文件名在Ollama的路径中又被执行了sha256散列变换。因此我们需要从索引文件中获取相应的文件名,再进…

阻塞IO 非阻塞IO

网络IO流程三次握手流程阻塞IO非阻塞IO阻塞IO和非阻塞IO的区别 阻塞IO:io 未就绪的情况下,会阻塞线程等待 非阻塞IO:io 未就绪的情况下,立即返回socket默认的情况是阻塞的

How Do Recommendation Models Amplify Popularity Bias? An Analysis from the Spectral Perspective

目录概符号说明Popularity bias\(\mathbf{q}_1\) 和 \(\mathbf{r}\) 具有高相似度相似度随着维度降低而增加相似度随着训练的变化ReSN: Regulartion with Spectral NormLin S., Gao C., Chen J., Zhou S., Hu B., Feng Y., Chen C. and Wang C. How do recommendation models a…

(文末有福利!)深度剖析大语言模型推理:指标、优化与框架选择

在人工智能飞速发展的当下,大语言模型(LLM)已然成为焦点。从智能聊天机器人到内容创作辅助,大语言模型的应用无处不在。但你是否了解其背后的推理过程,以及如何让这些模型运行得又快又好?今天,我们就来深入探讨大语言模型推理的奥秘。一、🌟大语言模型推理指标详解 在…

Docker之网络模型

Docker的网络模型类型 说明None 不为容器配置任何网络功能,没有网络 --net=noneContainer 与另一个运行中的容器共享Network Namespace,--net=container:containerID,k8s中Pod容器之间用此网络。Host 与主机共享Network Namespace,--net=hostBridge Docker设计的NAT网络模型…