38-二叉树练习-LeetCode145二叉树的后序遍历

news/2025/1/13 2:48:32/

题目

给你一棵二叉树的根节点 root ,返回其节点值的 后序遍历 。

示例 1:

输入:root = [1,null,2,3]
输出:[3,2,1]

示例 2:

输入:root = []
输出:[]

示例 3:

输入:root = [1]
输出:[1]

提示:

    树中节点的数目在范围 [0, 100] 内
    -100 <= Node.val <= 100

进阶:递归算法很简单,你可以通过迭代算法完成吗?


思路1:递归


代码1

/*** Definition for a binary tree node.* public class TreeNode {*     int val;*     TreeNode left;*     TreeNode right;*     TreeNode() {}*     TreeNode(int val) { this.val = val; }*     TreeNode(int val, TreeNode left, TreeNode right) {*         this.val = val;*         this.left = left;*         this.right = right;*     }* }*/
class Solution {List<Integer> ret = new LinkedList<>();public List<Integer> postorderTraversal(TreeNode root) {if(root == null) {return ret;}postorderTraversal(root.left);postorderTraversal(root.right);ret.add(root.val);return ret;}
}

思路2:


代码2

/*** Definition for a binary tree node.* public class TreeNode {*     int val;*     TreeNode left;*     TreeNode right;*     TreeNode() {}*     TreeNode(int val) { this.val = val; }*     TreeNode(int val, TreeNode left, TreeNode right) {*         this.val = val;*         this.left = left;*         this.right = right;*     }* }*/
class Solution {public List<Integer> postorderTraversal(TreeNode root) {List<Integer> ret = new LinkedList<>();if(root == null) {return ret;}Stack<TreeNode> stack = new Stack<>();//cur表示当前需要判断的节点TreeNode cur = root;//prev表示最近一次完全结束遍历的节点TreeNode prev = null;while(cur != null || !(stack.isEmpty())) {//一路向左走到头while(cur != null) {stack.push(cur);cur = cur.left;}//左树已经全部结束cur = stack.pop();//此时cur是第二次访问,不能直接遍历,需要判断右树是否为空或者是否遍历结束if(cur.right == null || prev == cur.right) {//右树已经结束遍历,第三次访问curret.add(cur.val);//此时cur已经全部访问完毕,prev更新为当前cur节点prev = cur;cur = null;} else {stack.push(cur);//继续访问右子树cur = cur.right;}}return ret;}
}

前、中、后序遍历借助栈实现,根节点啥时候能真正遍历输出?

  1. 前:第一次访问根节点就可以输出。
  2. 中:第二次访问根节点(走完左树)可以输出。
  3. 后:第三次访问根节点(走完子树)可以输出。

节点入栈和出栈:

  • 入栈:第一次走到节点入栈。
  • 出栈:当真正访问完这个节点出栈。(左、右、根)

http://www.ppmy.cn/news/37017.html

相关文章

雷达基础知识

雷达频率划分 以下是按照频率和波长划分雷达频段的表格&#xff1a; 波段名称频率范围&#xff08;GHz&#xff09;波长范围&#xff08;cm&#xff09;应用领域VHF0.03 - 0.3100 - 10气象雷达、空管雷达、航空雷达UHF0.3 - 3100 - 10航空雷达、海上雷达、地面雷达、火控雷达…

彻底关闭Windows更新

一、关闭Windows Update服务 1、按“Windows R”键&#xff0c;打开运行对话框&#xff0c;并输入“services.msc”&#xff0c;然后再单击“确定”。 2、在弹出的服务窗口中&#xff0c;找到“Windows Update”选项并双击打开它。 3、在弹出的“Windows Update的属性”对话框…

Springboot 多线程分批切割处理 大数据量List集合 ,实用示例

前言 哲学提问镇贴&#xff1a; 不了解异步怎么使用的看官&#xff0c; 可阅&#xff1a; SpringBoot 最简单的使用异步线程案例 Async_小目标青年的博客-CSDN博客 Springboot Async异步扩展使用 结合 CompletableFuture_小目标青年的博客-CSDN博客 想了解更多关于批量list处…

对象的封装

示例如下&#xff1a; package com.hy; /* 面向对象的特征一&#xff1a;封装与隐藏 一&#xff1a;问题的引入&#xff1a; 当我们创建一个类的对象以后&#xff0c;我们可以通过“对象.属性”的方式&#xff0c;对对象的属性进行赋值&#xff0c; 这里&#xff0c;赋值操作要…

Vue|计算属性

1. 计算属性1.1 差值语法1.2 methods1.3 计算属性1. 计算属性 1.1 差值语法 开始前分别在项目目录创建文件夹及页面如下 需求1&#xff1a;在两个文本框中分别输入姓和名的同时需要在下方将数据进行拼接组装&#xff0c;效果如下图 如果用传统的方式来实现的话&#xff0c;需要…

【Dockerfile学习笔记】常用命令及参数意义

什么是 Dockerfile&#xff1f; Dockerfile 是一个用来构建镜像的文本文件&#xff0c;文本内容包含了一条条构建镜像所需的指令和说明。 下面罗列一下常用的与镜像有关的命令&#xff1a; 列举本地的镜像&#xff1a;docker images 拉取镜像&#xff1a;docker pull tomcat …

线程池(2022-09-29)

文章目录前言一、线程池的常用构建方式1.构造方法实现(推荐)2.通过 Executor 框架的工具类 Executors 来实现(不推荐)3.Spring框架提供4.Guava&#xff0c;hutool的ThreadFactoryBuilder5.线程池的异常处理二、线程池核心 阻塞队列 BlockingQueue1.二级标题三、线程池异常1.try…

【理论推导】变分自动编码器 Variational AutoEncoder(VAE)

变分推断 (Variational Inference) 变分推断属于对隐变量模型 (Latent Variable Model) 处理的一种技巧&#xff0c;其概率图如下所示 我们将 X{x1,...xN}X\{ x_1,...x_N \}X{x1​,...xN​} 看作是每个样本可观测的一组数据&#xff0c;而将对应的 Z{z1,...,zN}Z\{z_1,...,z_N…