本文总结了刷LeetCode过程中,有关树的遍历的相关代码实现,包括了二叉树、N叉树先序、中序、后序、BFS、DFS遍历的递归和迭代实现。这也是解决树的遍历问题的固定套路。

一、二叉树的先序、中序、后序遍历
  1、递归模板
  (1)先序

1 public void preorder(TreeNode root) {
2 if (root == null) {
3 return;
4 }
5 res.add(root.val);
6 preorder(root.left);
7 preorder(root.right);
8 }

(2)中序

1 public void inorder(TreeNode root) {
2 if (root == null) {
3 return;
4 }
5 inorder(root.left);
6 res.add(root.val);
7 inorder(root.right);
8 }

(3)后序

1 public void postorder(TreeNode root) {
2 if (root == null) {
3 return;
4 }
5 postorder(root.left);
6 postorder(root.right);
7 res.add(root.val);
8 }

2、迭代模板:显式使用栈
  (1)先序 。也可以使用后文的DFS实现

 1 public void preorder(TreeNode root) {
2 Deque<TreeNode> stack = new ArrayDeque<>();
3 while(root !=null || !stack.isEmpty()) {
4 while(root != null) {
5 stack.push(root);
6 res.add(root.val);//保存结果
7 root = root.left;
8 }
9 root = stack.pop();
10 root = root.right;
11 }
12 }

(2)中序

 1 public void inorder(TreeNode root) {
2 Deque<TreeNode> stack = new ArrayDeque<>();
3 while(root != null || !stack.isEmpty()) {
4 while(root != null) {
5 stack.push(root);
6 root = root.left;
7 }
8 root = stack.pop();
9 res.add(root.val);//保存结果
10 root = root.right;
11 }
12 }

(3)后序。先序是根——左——右,而后序是左-右-根,可以将先序改成根-右-左,然后将结果反转。如下代码对比先序的代码:

 1 public void postorder(TreeNode root) {
2 Deque<TreeNode> stack = new ArrayDeque<>();
3 while(root != null || !stack.isEmpty()) {
4 while(root != null) {
5 stack.push(root);
6 res.add(root.val);//保存结果
7 root = root.right;//注意这里和先序、中序的差别
8 }
9 root = stack.pop();
10 root = root.left();
11 }
12 Collections.reverse(res);//将前面的结果反转
13 }

当然,上面这种方法比较间接,借助于先序遍历的理解。下面采用一种直接的迭代方式:

 1 public void postorder(TreeNode root) {
2 Deque<TreeNode> stack = new ArrayDeque<>();
3 TreeNode preNode = new TreeNode();//该节点用于保存前一个出栈的节点
4 while (root != null || !stack.isEmpty()) {
5 //将当前节点的左子树节点一次入栈
6 while (root != null) {
7 stack.push(root);
8 root = root.left;
9 }
10 root = stack.peek();
11 //当前节点没有右孩子了,或者其右孩子已经出栈了,则当前节点也出栈
12 if (root.right == null || root.right == preNode) {
13 root = stack.pop();
14 res.add(root.val);//保存结果
15 preNode = root; //记录本次刚输出的节点
16 root = null;
17 } else {
18 //当前节点还有右孩子,且其右孩子还没有出栈,则先处理其右孩子
19 root = root.right;
20 }
21 }
22 }

二、N叉树的先序与后序遍历
  1、递归模板
  (1)先序

 1 public void preorder(TreeNode root) {
2 if (root == null) {
3 return;
4 }
5 res.add(root.val);//保存结果
6 if (root.children != null) {
7 for (int i = 0; i < root.children.size(); i++) {
8 dfs(root.children.get(i));
9 }
10 }
11 }

(2)后序

1 public void postorder(TreeNode root) {
2 if (root == null) {
3 return;
4 }
5 for (int i = 0; i < root.children.size(); i++) {
6 postorder(root.children.get(i));
7 }
8 res.add(root.val);//保存结果
9 }

2、迭代模板
  (1)先序。即下文中DFS的实现

 1 public void preorder(Node root) {
2 Deque<Node> stack = new ArrayDeque<>(); //BFS使用队列,这里使用栈
3 stack.push(root);
4 while (!stack.isEmpty()) {
5 root = stack.pop();
6 res.add(root.val);//保存结果
7 int childCount = root.children == null ? 0 : root.children.size();
8 //这里需要注意孩子节点入栈的顺序
9 for (int i = childCount - 1; i >= 0; i--) {
10 stack.push(root.children.get(i));
11 }
12 }
13 }

(2)后序。先序是根——左——右,而后序是左-右-根,可以将先序改成根-右-左,然后将结果反转。如下代码对比先序的代码:

 1 public void postorder(Node root) {
2 List<Integer> result = new ArrayList<>();
3 Deque<Node> stack = new ArrayDeque<>();
4 stack.push(root);
5 while (!stack.isEmpty()) {
6 root = stack.pop();
7 result.add(root.val);//保存结果
8 int childCount = root.children == null ? 0 : root.children.size();
9 //还入栈的顺序正好和先序遍历相反
10 for (int i = 0; i < childCount; i++) {
11 stack.push(root.children.get(i));
12 }
13 }
14 //将结果反转
15 Collections.reverse(result);
16 for (Integer i:result) {
17 System.out.print(i);
18 }
19 }

目前还没有找到类似二叉树直接后序遍历的方法

三、树的BFS(广度优先搜索)和DFS(深度优先搜索)实现模板
  1、递归实现
  (1)BFS
          一般来说不能使用递归来实现BFS,因为BFS使用的时队列实现,而递归使用的是栈实现。
  (2)DFS
          普通的N叉树的DFS包括先序遍历和后序遍历,它们的递归实现和前文一致。如果是二叉树,还有中序遍历,递归实现和前文一致。
  2、迭代实现
  (1)BFS。即按层次来遍历

 1 public void bfs(Node root) {
2 Queue<Node> queue = new LinkedList<>();
3 queue.offer(root);
4 while (!queue.isEmpty()) {
5 root = queue.poll();
6 res.add(root.val);//保存结果
7 if (root.children != null) {
8 queue.addAll(root.children);
9 }
10 }
11 }

(2)DFS
       先序遍历、中序遍历(二叉树)和后序遍历的迭代方式实现和前文一致。

四、图的BFS和DFS遍历模板

树是图的一种特殊情况,相比于树而言图中可能出现环,所以在遍历的时候可能重复访问。所以树的BFS和DFS实现需要在树的基础上维护一个Set来避免访问重复的节点即可。

1、BFS

 1 public void graphyBfs(Node root) {
2 if (root == null) {
3 return;
4 }
5 Set<Node> visitedSet = new HashSet<>(); //维护一个set,保存已经访问过的节点
6 Queue<Node> queue = new LinkedList<>();
7 queue.offer(root);
8 while (!queue.isEmpty()) {
9 root = queue.poll();
10 visitedSet.add(root);//保存每个已经输出的节点
11 res.add(root.val);//保存结果
12 if (root.children == null) {
13 return;
14 }
15 for (int i = 0; i < root.children.size(); i++) {
16 Node tmpNode = root.children.get(i);
17 if (!visitedSet.contains(tmpNode)) {//已经输出过的节点,则不用再加入
18 queue.offer(tmpNode);
19 }
20 }
21 }
22 }

2、DFS
       迭代方式、递归方式,都维护一个set来保存已经访问过的节点,在输出节点时保存到set中,在添加节点时添加去重操作即可。

【算法】二叉树、N叉树先序、中序、后序、BFS、DFS遍历的递归和迭代实现记录(Java版)的更多相关文章

  1. LeetCode:二叉树的前、中、后序遍历

    描述: ------------------------------------------------------- 前序遍历: Given a binary tree, return the pr ...

  2. TZOJ 3209 后序遍历(已知中序前序求后序)

    描述 在数据结构中,遍历是二叉树最重要的操作之一.所谓遍历(Traversal)是指沿着某条搜索路线,依次对树中每个结点均做一次且仅做一次访问. 这里给出三种遍历算法. 1.中序遍历的递归算法定义:  ...

  3. [Java]算术表达式求值之二(中序表达式转后序表达式方案,支持小数)

    Inlet类,入口类,这个类的主要用途是验证用户输入的算术表达式: package com.hy; import java.io.BufferedReader; import java.io.IOEx ...

  4. [Java]算术表达式求值之一(中序表达式转后序表达式方案)

    第二版请见:https://www.cnblogs.com/xiandedanteng/p/11451359.html 入口类,这个类的主要用途是粗筛用户输入的算术表达式: package com.h ...

  5. DS Tree 已知先序、中序 => 建树 => 求后序

    参考:二叉树--前序和中序得到后序 思路历程: 在最初敲的时候,经常会弄混preorder和midorder的元素位置.大体的思路就是在preorder中找到根节点(根节点在序列的左边),然后在mid ...

  6. 已知树的前序、中序,求后序的java实现&已知树的后序、中序,求前序的java实现

    public class Order { int findPosInInOrder(String str,String in,int position){ char c = str.charAt(po ...

  7. 分别求二叉树前、中、后序的第k个节点

    一.求二叉树的前序遍历中的第k个节点 //求先序遍历中的第k个节点的值 ; elemType preNode(BTNode *root,int k){ if(root==NULL) return ' ...

  8. HDU 1710 (二叉树的前序和中序,求后序)

    题目链接 题目大意: 输入二叉树的前序.中序遍历,请输出它的后序遍历 #include <stdio.h> #include <string.h> ; // 长度为n s1 前 ...

  9. hdu1710-Binary Tree Traversals (由二叉树的先序序列和中序序列求后序序列)

    http://acm.hdu.edu.cn/showproblem.php?pid=1710 Binary Tree Traversals Time Limit: 1000/1000 MS (Java ...

随机推荐

  1. PYG5.4第十六期第一轮基础六题

    1. HYWZ-dts音效大师破解https://www.chinapyg.com/thread-135090-1-1.html(出处: 飘云阁(PYG官方论坛) ) 2. HYWZ-LopeEdit ...

  2. 数据库图形表Navicat Premium

    1.什么是数据库? 存储数据,为了方便查询和使用 web时代使用最广泛的关系型数据库 2.历史: 瑞典公司开发,卖给SUN,SUN又卖给ORACLE 开源,免费,支持多平台 3.数据库图形表Navic ...

  3. ios自动识别电话并变色的问题解决方法

    问题: 在做移动端页面时发现长串数字都被ios系统的手机识别为电话号码,且文字变成很土的蓝色,点击有下划线并弹出提示拨打该电话号码. 解决方法: 1.在head中加上下面这行代码就OK了(仅限于单页面 ...

  4. 摊牌了!我要手写一个“Spring Boot”

    目前的话,已经把 Spring MVC 相关常用的注解比如@GetMapping .@PostMapping .@PathVariable 写完了.我也已经将项目开源出来了,地址:https://gi ...

  5. 02 C语言最简单快速上手的IDE

    不要让开发环境 成为学习路上的拦路虎,先培养学习兴趣和学习路上的成就感,这个对于激励自己持续学习很重要. 等真正从小白入门了,甚至成为大牛了,能解决诸多困难问题了,安装个开发环境还会再是个什么难事吗? ...

  6. visual studio 2015 安装MSDN全称Microsoft Developer Network 安装离线的MSDN

    MSDN: 微软向开发人员提供的一套帮助系统,其中包含大量的开发文档,技术文章和示例代码. 这里介绍了vs2015 装离线的MSDN(说明一点是,如果不行,说明你的文件有缺陷,没安装好,之前我用vs2 ...

  7. Spring的BeanFactory是什么?

    什么是BeanFactory? 提到Spring,总是让人第一时间想起IOC容器,而IOC容器的顶层核心接口就是我们的BeanFactory,如果能够理解BeanFactory的体系结构想必能让我们对 ...

  8. XML节点自动生成简单实例

    有些时候我们在拼装XML的过程中,因为各种拼接会感到非常的麻烦(定义变量模型,自动生成,使用XElement再去组装),我的脑袋感觉都大了,能不能有个稍微比较快捷自动随变量自动生成XML格式的方式,看 ...

  9. 跨时代的MySQL8.0新特性解读

    目录 MySQL发展历程 MySQL8.0新特性 秒级加列 性能提升 文档数据库 SQL增强 共用表表达式(CTEs) 不可见索引(Invisible Indexes) 降序索引(Descending ...

  10. Python库之SQLAlchemy

    一.SQLAlchemy简介 1.1.SQLAlchemy是什么? sqlalchemy是一个python语言实现的的针对关系型数据库的orm库.可用于连接大多数常见的数据库,比如Postges.My ...