【刷爆力扣之589-590. N叉树的前序遍历和后序遍历】

589:N叉树的前序遍历

这道题需要灵活的理解前序遍历的规则,从二叉树到N叉树,原则都是统一的,因此我们可以借鉴二叉树的前序遍历方式,使用递归 以及迭代两种方式完成N叉树的前序遍历

方法一:递归

思路:

递归思路比较简单,N 叉树的前序遍历与二叉树的前序遍历的思路和方法基本一致,可以参考「144. 二叉树的前序遍历」的方法,每次递归时,先访问根节点,然后依次递归访问每个孩子节点即可。

java 复制代码
public List<Integer> preorder(Node root) {
    List<Integer> res = new ArrayList<>();
    // 调用递归函数
    doPreorder(root, res);
    return res;
}

private void doPreorder(Node node, List<Integer> list) {
    if (node == null) {
        return;
    }
    list.add(node.val);
    // 遍历当前节点的子节点,递归执行
    for (Node child : node.children) {
        doPreorder(child, list);
    }
}

方法二:迭代

思路:

在前序遍历中,我们会先遍历节点本身,然后从左向右依次先序遍历该每个以子节点为根的子树,此时利用 先进后出的原理,依次从右向左将子节点入栈,这样出栈的时候即可保证从左向右依次遍历每个子树。

首先把根节点入栈,因为根节点是前序遍历中的第一个节点。随后每次我们从栈顶取出一个节点 u,它是我们当前遍历到的节点,并把 u 的所有子节点从右向左逆序压入栈中,这样出栈的节点则是顺序从左向右的。

java 复制代码
public List<Integer> preorder(Node root) {
    List<Integer> res = new ArrayList<>();
    if (root == null) {
        return res;
    }
    // 前序遍历需要的栈数据结构
    Stack<Node> stack = new Stack<>();
    stack.push(root);
    while (!stack.isEmpty()) {
        Node pop = stack.pop();
        res.add(pop.val);
        if (!pop.children.isEmpty()) {
            // 当前节点的孩子需要从后往前入栈,才能从前往后出栈实现前序遍历
            // 从右向左逆序压入栈中
            for (int i = pop.children.size() - 1; i >= 0; i--) {
                stack.push(pop.children.get(i));
            }
        }
    }
    return res;
}

590:N叉树的后序遍历

方法一:递归

思路:

递归思路比较简单,NNN 叉树的前序遍历与二叉树的后序遍历的思路和方法基本一致,可以参考「145. 二叉树的后序遍历」的方法,每次递归时,先递归访问每个孩子节点,然后再访问根节点即可。

java 复制代码
// 递归实现
public List<Integer> postorder(Node root) {
    List<Integer> res = new ArrayList<>();
    doPostorder(root, res);
    return res;
}

private void doPostorder(Node node, List<Integer> res) {
    if (node == null) {
        return;
    }
    node.children.forEach(child -> {
        doPostorder(child, res);
    });
    res.add(node.val);
}

方法二:迭代

思路:

在后序遍历中,我们会先从左向右依次后序遍历每个子节点为根的子树,再遍历根节点本身。此时利用栈先进后出的原理,依次从右向左将子节点入栈,这样出栈的时候即可保证从左向右依次遍历每个子树。

首先把根节点入栈,因为根节点是前序遍历中的第一个节点。随后每次我们找到栈顶节点 u,如果当前节点的子节点没有遍历过,则应该先把 u 的所有子节点从右向左逆序压入栈 中,这样出栈的节点则是顺序从左向右的,同时对节点 u 进行标记,表示该节点的子节点已经全部入栈;如果当前节点 u 为叶子节点或者当前节点的子节点已经全部遍历过,则从栈中弹出节点 u,并记录节点 uuu 的值。

java 复制代码
// 迭代实现
public List<Integer> postorder2(Node root) {
    List<Integer> res = new ArrayList<>();
    if (root == null) {
        return res;
    }
    Stack<Node> stack = new Stack<>();
    // 标记节点是否被处理
    List<Node> visited = new ArrayList<>();
    stack.push(root);
    while (!stack.isEmpty()) {
        Node peek = stack.peek();
        // 如果当前节点没有孩子,或者当前节点已经被标记(说明子节点已经处理完了,可以处理当前节点了)
        if (peek.children.isEmpty() || visited.contains(peek)) {
            Node pop = stack.pop();
            res.add(pop.val);
            continue;
        }
        for (int i = peek.children.size() - 1; i >= 0; i--) {
            stack.push(peek.children.get(i));
            // 标记当前节点的所有子节点已经全部入栈
            visited.add(peek);
        }
    }
    return res;
}

方法三:利用前序遍历反转

前序遍历为中 => 左 => 右

后序遍历为 左 => 右 => 中

因此,我们可以使用前序遍历先构造一个 中 => 右 => 左 的结果集合,然后将该结果集合翻转得到后序遍历的结果,那么正常的前序遍历 中 => 左 => 右 怎么得到 中 => 右 => 左 的结果集合呢?

其实, 中 => 左 => 右 的遍历方式是因为我们在将中间节点的左右节点压栈时使用的是先压入右节点,再压入左节点的方式 ,那么我们想要得到 中 => 右 => 左 的结果集合,只需要在压入中间节点左右节点时先压入左节点,再压入右节点即可!

java 复制代码
// 利用前序遍历反转
public List<Integer> postorder3(Node root) {
    List<Integer> res = new ArrayList<>();
    if (root == null) {
        return res;
    }
    Stack<Node> stack = new Stack<>();
    stack.push(root);
    while (!stack.isEmpty()) {
        Node pop = stack.pop();
        res.add(pop.val);
        if (!pop.children.isEmpty()) {
            // 先压入左节点,再压入右节点
            for (Node child : pop.children) {
                stack.push(child);
            }
        }
    }
    // 翻转得到后序遍历的结果
    Collections.reverse(res);
    return res;
}

部分解释来源于力扣题解

. - 力扣(LeetCode)

代码均为原创!

相关推荐
巫师不要去魔法部乱说1 小时前
PyCharm专项训练4 最小生成树算法
算法·pycharm
IT猿手1 小时前
最新高性能多目标优化算法:多目标麋鹿优化算法(MOEHO)求解GLSMOP1-GLSMOP9及工程应用---盘式制动器设计,提供完整MATLAB代码
开发语言·算法·机器学习·matlab·强化学习
阿七想学习1 小时前
数据结构《排序》
java·数据结构·学习·算法·排序算法
王老师青少年编程1 小时前
gesp(二级)(12)洛谷:B3955:[GESP202403 二级] 小杨的日字矩阵
c++·算法·矩阵·gesp·csp·信奥赛
Kenneth風车2 小时前
【机器学习(九)】分类和回归任务-多层感知机(Multilayer Perceptron,MLP)算法-Sentosa_DSML社区版 (1)111
算法·机器学习·分类
eternal__day2 小时前
数据结构(哈希表(中)纯概念版)
java·数据结构·算法·哈希算法·推荐算法
APP 肖提莫2 小时前
MyBatis-Plus分页拦截器,源码的重构(重构total总数的计算逻辑)
java·前端·算法
OTWOL2 小时前
两道数组有关的OJ练习题
c语言·开发语言·数据结构·c++·算法
qq_433554543 小时前
C++ 面向对象编程:递增重载
开发语言·c++·算法
带多刺的玫瑰3 小时前
Leecode刷题C语言之切蛋糕的最小总开销①
java·数据结构·算法