算法通关村—迭代实现二叉树的前序,中序,后序遍历

news/2024/10/22 9:52:14/

1. 前序中序后序递归写法

前序

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

后序

public static void postOrderRecur(TreeNode head) {if (head == null) {return;}postOrderRecur(head.left);postOrderRecur(head.right);System.out.print(head.value + " ");
}

中序

public static void inOrderRecur(TreeNode head) {if (head == null) {return;}inOrderRecur(head.left);System.out.print(head.value + " ");inOrderRecur(head.right);
}

2. 前序遍历迭代法

前序遍历的主要特征是中左右
在这里插入图片描述
上面的前序遍历是:1 2 4 5 3 6 7
很明显 1 2 4都是左子树,然后遍历完了才到右子树,那么迭代需要做的就是一直遍历左子树节点,然后保存当前的左子树的根节点,左子树完了,然后去除节点找到右节点。这里面需要使用栈来进行存储节点,然后按照相应的顺序弹出节点。

2.1 二叉树的前序遍历

二叉树的前序遍历
给你二叉树的根节点 root ,返回它节点值的 前序 遍历。
输入:root = [1,null,2,3]
输出:[1,2,3]
在这里插入图片描述

 public List<Integer> preorderTraversal(TreeNode root) {List<Integer> res = new ArrayList<>();if(root == null)  return res;Deque<TreeNode> stack = new LinkedList<>();TreeNode node = root;while(!stack.isEmpty() || node!=null){while(node!=null){res.add(node.val);// 保存根节点,找到对应的右节点stack.push(node);// 处理左子树node = node.left;}// 找到对应的右节点的根节点node = stack.pop();// 处理右子树node=node.right;}return res;}

3. 迭代法中序遍历

特点:左中右
在这里插入图片描述
元素:4 2 5 1 6 3 7
这个可以看作每次先遍历最左的子树节点,然后输出最下面的节点,逆序,如果根节点还有右节点,就继续进入右节点到最下面,否则依然逆序输出。依然需要使用一个栈来存储这个节点。

3.1 二叉树的中序遍历

二叉树的中序遍历
给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。

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

总体上一致,只不过添加的时候是添加的当前链路最后一个节点元素。

    public List<Integer> inorderTraversal(TreeNode root) {List<Integer> res = new ArrayList<>();if(root ==null) return res;Deque<TreeNode> stack = new LinkedList<>();TreeNode node = root;while(!stack.isEmpty() || node!=null){while(node!=null){stack.push(node);node=node.left;}// 此时已经到头了node = stack.pop();res.add(node.val);node=node.right;}return res;}

4. 后序遍历

特点:左右中
在这里插入图片描述
元素:4 5 2 6 7 3 1
元素特点:需要输出当前根节点的两个子节点的值,然后再输出根节点。但是实际操作下来发现很麻烦,将后序便利的元素反转变成 1 3 7 6 2 5 4, 而这样就和前序类似,只不过区别在于前序先遍历左节点,现在需要遍历右节点,然后将列表元素整体反转。

4.1 二叉树的后序遍历

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

 public List<Integer> postorderTraversal(TreeNode root) {List<Integer> res = new ArrayList<>();if(root == null) return res;Deque<TreeNode> stack = new LinkedList<>();TreeNode node = root;while(!stack.isEmpty() || node!=null){while(node!=null ){res.add(node.val);stack.push(node);node= node.right;}node =  stack.pop();node = node.left;}Collections.reverse(res);return res;}

但是这个方法并没有用到相关的后序遍历的特性,只是使用的前序

4.2 迭代法

public List<Integer> postorderTraversal(TreeNode root) {List<Integer> res = new ArrayList<>();if(root == null) return res;Deque<TreeNode> stack = new LinkedList<>();TreeNode node = null;while(!stack.isEmpty() || root!=null){while(root!=null ){stack.push(root);root= root.left;}root =  stack.pop();// 如果右子树为空或者已经被访问过了,才能添加当前节点值if(root.right==null || root.right == node){res.add(root.val);node=root;root=null;}else{stack.push(root);root = root.right;}}return res;}

这个方法有一点难以理解,但是也能看得懂相关步骤。


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

相关文章

20230803 linux信号量sem sem_init(sem_t* m_sem,0,0)

信号量及一切定义为指针类型的变量使用前一定要先new 一个实例化对象将地址给该指针&#xff0c;否则指针没有确定的地址&#xff0c;运行后直接访问该错误地址报段错误。 信号量使用参考&#xff1a; linux 多线程之信号量 sem_init 有名信号量sem_open和内存信号量sem_init…

5、二叉树

二叉树遍历 递归序 public static void f(Node head) {if (head == null) {return;}f(head.left);f(head.right); }前中后遍历_递归 public static void preOrderRecur(Node head) {if (head == null) {return;}System.out.print(head.value + " ");preOrderRecur…

【EI/SCOPUS征稿】第三届智能电网与能源互联网国际会议(SGEI 2023)

第三届智能电网与能源互联网国际会议&#xff08;SGEI 2023&#xff09; 2023 3rd International Conference on Smart Grid and Energy Internet 为交流近年来国内外在智能电网和能源互联网领域的理论、技术和应用的最新进展&#xff0c;展示最新成果&#xff0c;2023年第三…

宋浩高等数学笔记(十)重积分

本章更新第10章重积分&#xff0c;关于三重积分的应用部分暂时略过&#xff0c;本部分在考察的时候不会很难&#xff0c;困难在于对重积分本质的理解&#xff0c;以及极坐标下相关公式的计算。类比普通的定积分&#xff0c;如果对一个宽度不均匀的函数&#xff0c;求积分分后相…

运维项目—K8S命令

文章目录 一、基本操作1、命名空间kubectl get ns 获取命名空间kubectl get ns default -o yaml 以yaml的格式查看某个nskubectl describe ns hoc-prod 查看某个ns详情1、命名空间与Podkubectl get pods --all-namespaces查看所有命名空间下的所有podkubectl get pod -A查看所有…

浅谈编程语言的函数与方法

在编程中&#xff0c;函数&#xff08;Function&#xff09;和方法&#xff08;Method&#xff09;是非常重要的概念&#xff0c;都是在编程中用来执行特定功能的代码块&#xff0c;可以被调用或重复使用&#xff0c;从而提高代码的可读性&#xff0c;可维护性和重用性。 函数&…

redis 如何保证数据一致性

前言 日常开发中常会使用redis作为项目中的缓存&#xff0c;只要我们使用 Redis 缓存&#xff0c;就必然会面对缓存和数据库间的一致性保证问题。而且如果数据不一致&#xff0c;那么应用从缓存中读取的数据就不是最新数据&#xff0c;可能会导致严重的业务问题。 为什么会数…

2023年华数杯赛题浅析

2023年华数杯作为与国赛同频的比赛&#xff08;周四6点发题&#xff0c;周日晚8点交卷&#xff09;&#xff0c;也是暑期唯一一个正式比赛。今年的报名队伍已经高达​6000多对。基于这么多的人数进行国赛前队伍的练习&#xff0c;以及​其他用途。为了方便大家跟更好的选题&…