IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 145. 二叉树的后序遍历(迭代遍历) -> 正文阅读

[数据结构与算法]145. 二叉树的后序遍历(迭代遍历)

145. 二叉树的后序遍历

给定一个二叉树,返回它的 后序 遍历。

示例:

输入: [1,null,2,3]  
   1
    \
     2
    /
   3 

输出: [3,2,1]

对于后序遍历,由于遍历的顺序是左->右->中,所以某个节点在其左子树遍历完成后并不能立即出栈,而是要等待其右子树叶遍历完成后再出栈,那么我们如何知道当前这次出栈是因为左子树遍历完了还是右子树遍历完了呢?

所以我们要借助一种新的数据结构来辅助记录该节点是第一次出栈还是第二次。如下

 class TagNode{
      TreeNode node;
      boolean flag;

      public TagNode(){}
      public TagNode(TreeNode node ,boolean flag){
         this.node = node;
         this.flag = flag;
      }
  }

可以看到TagNode包含两个字段,一个是树的节点,一个是标志字段flag,刚开始flag字段初始化为false,

如果因为当前节点的左子树遍历完成而导致当前节点出栈时,我们将这个flag字段的值更新为true,然后再将其压入栈中,然后等到其右子树遍历完成后,该节点出栈时,检查标志位flag的值为true,才可以真正的出栈。

举例说明二叉树的后序遍历的遍历过程。

image-20211003222131674

为1节点新创建一个TagNode,flag为false,并入栈

image-20211003222357057

随后指针移动到1节点的子节点,发现为null。

image-20211003222520382

栈顶元素出栈,检查flag标志为发现为false,表示这是第一次出栈,我们将flag的值改为true再将这个节点压回栈中,转而遍历其右子树。

image-20211003223148292

然后为2节点创建一个TagNode flag设置为false,压入栈中

image-20211003223210570

递归遍历左子树,直至为null

image-20211003223245721

image-20211003223304963

此时now指针为null,栈顶元素出栈,检查flag值为false,将flag值变为true后再次压入栈中,转而遍历其右子树。

image-20211003223429692

然后发现,其右子节点的值同样是null,该节点再次出栈,检查其flag值为true,符合出栈要求,将其值加入res数组后将其出栈。

image-20211003223552217

此时,now依然指向null,所以我们仍然需要从栈中出栈一个元素,所以由将2节点出栈,检查其flag值为flase,将flag值变为true后再次压入栈中,转而遍历其右子树。

image-20211003223744887

但不幸的是,此时2节点的右子树依然为null,所以我们再次将栈顶节点 2节点出栈,检查其flag值为true,符合出栈要求,将其值加入res数组后,出栈该节点。

image-20211003223922345

此时,now指针依然指向null,所以需要从栈中再次出栈一个节点,这个节点是1节点,检查其flag值为true,符合出栈要求,将其值加入res数组后,将该节点出栈。

image-20211003224032657

此时,now指向null且栈中无任何可出栈元素,遍历过程结束,得到所求的数组res。

代码如下:

/**
 * 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 {
  static class TagNode{
      TreeNode node;
      boolean flag;

      public TagNode(){}
      public TagNode(TreeNode node ,boolean flag){
         this.node = node;
         this.flag = flag;
      }
  }
  public List<Integer> postorderTraversal(TreeNode root) {
      List<Integer> res = new ArrayList<>();
      if(root == null){return res;}

      Deque<TagNode> stack = new ArrayDeque<>();
      TreeNode now = root;
      TagNode node ;
      while(now != null || !stack.isEmpty()){
        while(now != null){
          node = new TagNode(now,false);
          stack.push(node);
          now = now.left;
        }

        node = stack.pop();
        if(node.flag == true){
          res.add(node.node.val);
        }else{
          node.flag = true;
          stack.push(node);
          now = node.node.right;
        }
      }
      return res;
    }
}

总结:

  • 进易出难:对于某个节点,只要now指针指向它就立刻将其入栈,而如果想将一个节点出栈,那么必须要两次检查flag值,满足条件才可以出栈。
  • 如果出栈节点的flag节点值为false,说明其右子树还未被遍历,更新flag值后重新压入栈中,now指针转而指向其右子树
  • 如果出栈节点的flag节点值为true,说明其左右子树均已被遍历,满足出栈条件。
  • 内部类要用static表示为静态类。
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-10-04 13:05:02  更:2021-10-04 13:06:23 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/4 16:51:14-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码