前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【小Y学算法】⚡️每日LeetCode打卡⚡️——40.二叉树的后序遍历

【小Y学算法】⚡️每日LeetCode打卡⚡️——40.二叉树的后序遍历

作者头像
呆呆敲代码的小Y
发布2021-09-26 11:13:19
2170
发布2021-09-26 11:13:19
举报
文章被收录于专栏:呆呆敲代码的小Y 公众号


????前言

  • ????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????
  • ???? 每天打卡一道算法题,既是一个学习过程,又是一个分享的过程????
  • ???? 提示:本专栏解题 编程语言一律使用 C# 和 Java 两种进行解题
  • ???? 要保持一个每天都在学习的状态,让我们一起努力成为算法大神吧????!
  • ???? 今天是力扣算法题持续打卡第40天????!
  • ????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????????

????原题样例:二叉树的后序遍历

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

示例 1:

代码语言:javascript
复制
输入: [1,null,2,3]  
   1
    \
     2
    /
   3 

输出: [3,2,1]

????C#方法一:迭代

仅使用一个栈完成后续遍历。步骤如下:

  1. 一直往左节点访问;
  2. 按根节点->右节点->左节点的顺序将节点依次压入栈中;
  3. 碰到叶子节点即开始出栈;
  4. 如果当前栈顶元素指向的节点依然有未访问到的子节点则回到步骤1,往复循环;
  5. 栈空,则遍历结束。

难点在于如何判断栈顶节点是否有未访问的子节点。 如果判断方式不当,很可能会因为栈顶节点是上一个已出栈节点的父节点,而导致其节点反复入栈出栈陷入死循环。

可以增加一个变量或指针,用于指向前一个已经出栈的节点。

每次判断栈顶节点的时候只需要提前判断一次是否与上一个出栈节点是父子关系,是的话则继续出栈,否则回到上面的步骤1再进行入栈循环。

代码语言:javascript
复制
public class Solution {
    public IList<int> PostorderTraversal(TreeNode root) {
        List<int> rst = new List<int>();
        if(root == null) return rst;

        Stack<TreeNode> st = new Stack<TreeNode>();
        TreeNode node = root;
        TreeNode printNode = root;
        TreeNode tmp = null;
        st.Push(node);
        while(st.Any())
        {
            tmp = null;
            if(node.left != printNode && node.right != printNode)
            {
                if(node.right != null)
                {
                    st.Push(node.right);
                    tmp = node.right;
                }
                if(node.left != null)
                {
                    st.Push(node.left);
                    tmp = node.left;
                }
            }
            node = tmp;
            if(tmp == null)
            {
                rst.Add((printNode = st.Pop()).val);
                if(st.Any())
                    node = st.Peek();
            }
        }
        return rst;
    }
}

执行结果

执行通过,执行用时284ms,内存消耗30.1MB.


????Java方法一:递归

思路和算法 首先我们需要了解什么是二叉树的后序遍历:按照访问左子树——右子树——根节点的方式遍历这棵树,而在访问左子树或者右子树的时候,我们按照同样的方式遍历,直到遍历完整棵树。 因此整个遍历过程天然具有递归的性质,我们可以直接用递归函数来模拟这一过程。

定义 postorder(root) 表示当前遍历到 root 节点的答案。 按照定义,我们只要递归调用 postorder(root->left) 来遍历 root 节点的左子树,然后递归调用 postorder(root->right) 来遍历 root 节点的右子树,最后将 root 节点的值加入答案即可,递归终止的条件为碰到空节点。

代码语言:javascript
复制
class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<Integer>();
        postorder(root, res);
        return res;
    }

    public void postorder(TreeNode root, List<Integer> res) {
        if (root == null) {
            return;
        }
        postorder(root.left, res);
        postorder(root.right, res);
        res.add(root.val);
    }
}

执行结果 执行通过

代码语言:javascript
复制
执行用时 0 ms, 在所有 Java 提交中击败了100.00%的用户
内存消耗 36.6 MB, 在所有 Java 提交中击败了68.49%的用户
????Java方法二:迭代

总体思路 我们也可以用迭代的方式实现方法一的递归函数,两种方式是等价的,区别在于递归的时候隐式地维护了一个栈,

而我们在迭代的时候需要显式地将这个栈模拟出来,其余的实现与细节都相同,具体可以参考下面的代码。

代码语言:javascript
复制
class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<Integer>();
        if (root == null) {
            return res;
        }

        Deque<TreeNode> stack = new LinkedList<TreeNode>();
        TreeNode prev = null;
        while (root != null || !stack.isEmpty()) {
            while (root != null) {
                stack.push(root);
                root = root.left;
            }
            root = stack.pop();
            if (root.right == null || root.right == prev) {
                res.add(root.val);
                prev = root;
                root = null;
            } else {
                stack.push(root);
                root = root.right;
            }
        }
        return res;
    }
}

执行结果

执行结果 执行通过

代码语言:javascript
复制
执行用时 0 ms, 在所有 Java 提交中击败了100.00%的用户
内存消耗 36.7 MB, 在所有 Java 提交中击败了40.52%的用户

????总结

  • 今天是力扣算法题打卡的第四十天!
  • 文章采用 C# 和 Java 两种编程语言进行解题
  • 一些方法也是参考力扣大神写的,也是边学习边分享,再次感谢算法大佬们
  • 那今天的算法题分享到此结束啦,明天再见!
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021/09/24 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • ????前言
  • ????原题样例:二叉树的后序遍历
    • ????C#方法一:迭代
      • ????Java方法一:递归
        • ????Java方法二:迭代
        • ????总结
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档