时间:2021-05-19
前序(先序)遍历
中序遍历
后续遍历
层序遍历
如图二叉树:
二叉树结点结构
public class TreeNode { int val; TreeNode left; TreeNode right; TreeNode(int x){ val=x; } @Override public String toString(){ return "val: "+val; }}访问函数
public void visit(TreeNode node){ System.out.print(node.val+" "); }前序遍历
对于图中二叉树而言其前序遍历结果为:6 2 0 1 4 5 8 9
二叉树的前序遍历即先遍历根结点再遍历左结点最后遍历右结点,使用递归如下:
非递归:
利用栈来实现二叉树的先序非递归遍历
/** * 非递归先序遍历二叉树 * */ public List<Integer> preorderTraversal(TreeNode root) { List<Integer> resultList=new ArrayList<>(); Stack<TreeNode> treeStack=new Stack<>(); if(root==null) //如果为空树则返回 return resultList; treeStack.push(root); while(!treeStack.isEmpty()){ TreeNode tempNode=treeStack.pop(); if(tempNode!=null){ resultList.add(tempNode.val);//访问根节点 treeStack.push(tempNode.right); //入栈右孩子 treeStack.push(tempNode.left);//入栈左孩子 } } return resultList; }更新:评论里有人说不理解非递归的先序遍历,其实你举个例子,然后画个图就可以理解了,以上图中的二叉树为例,先将6入栈,此时List为空,Stack只有一个元素6,进入while循环,弹出栈顶加入List,将6的右孩子和左孩子入栈,此时Stack从栈底到栈顶元素为8,2,List元素为6,由于栈不为空,进入while循环,弹出栈顶2,将2加入List,同时将2的右孩子和左孩子分别入栈,此时Stack从栈底到栈顶的元素为8,4,0, List的元素为6,2,由于栈不为空再次进入while循环…依次下去,弹出0加入List,入栈1,null,此时Stack从栈底到栈顶为8,4,1,null,List为6,2,0,弹出null为空继续弹出1,如此下去就可以了…
中序遍历
对于二叉树的中序遍历,即先访问左结点再访问根节点最后访问右结点
递归方法如下:
/** * 递归中序遍历 * */ public void preOrderRecursion(TreeNode node){ if(node==null) //如果结点为空则返回 return; preOrderRecursion(node.left);//访问左孩子 visit(node);//访问根节点 preOrderRecursion(node.right);//访问右孩子 }非递归:
在上图中的二叉树,其中序遍历为:0 1 2 4 5 6 8 9
可以看到,二叉树的中序遍历如下:
先将根节点入栈,
一直往其左孩子走下去,将左孩子入栈,直到该结点没有左孩子,则访问这个结点,如果这个结点有右孩子,则将其右孩子入栈,重复找左孩子的动作,这里有个要判断结点是不是已经被访问的问题。
非递归中序遍历(效率有点低),使用map(用set貌似更合理)来判断结点是否已经被访问
Discuss中有人给出更简洁的方法
public List<Integer> inorderTraversal(TreeNode root) { List<Integer> list = new ArrayList<Integer>(); Stack<TreeNode> stack = new Stack<TreeNode>(); TreeNode cur = root; while(cur!=null || !stack.empty()){ while(cur!=null){ stack.add(cur); cur = cur.left; } cur = stack.pop(); list.add(cur.val); cur = cur.right; } return list;}后序遍历
递归代码就不贴了
如果之前的非递归中序遍历使用map的方法理解后,后序遍历的话我们也可以使用一个map来保存那些已经被访问的结点,后序遍历即先访问左孩子再访问右孩子最后访问根结点。
非递归代码:
Discuss中有人给出了一个”巧“的方法,即先采用类似先序遍历,先遍历根结点再右孩子最后左孩子(先序是先根结点再左孩子最后右孩子),最后把遍历的序列逆转即得到了后序遍历
public List<Integer> postorderTraversal(TreeNode root) { Deque<TreeNode> stack = new LinkedList<>(); stack.push(root); List<Integer> ret = new ArrayList<>(); while (!stack.isEmpty()) { TreeNode node = stack.pop(); if (node != null) { ret.add(node.val); stack.push(node.left); stack.push(node.right); } } Collections.reverse(ret); return ret;}层序遍历
层序遍历也即宽度优先搜索,一层一层搜索,非递归代码如下:
public List<List<Integer>> levelOrder(TreeNode root) { List<List<Integer>> resultList=new ArrayList<>(); int levelNum=0;//记录某层具有多少个节点 Queue<TreeNode> treeQueue=new LinkedList<>(); treeQueue.add(root); while(!treeQueue.isEmpty()){ levelNum=treeQueue.size(); List<Integer> levelList=new ArrayList<>(); while(levelNum>0){ TreeNode tempNode=treeQueue.poll(); if(tempNode!=null){ levelList.add(tempNode.val); treeQueue.add(tempNode.left); treeQueue.add(tempNode.right); } levelNum--; } if(levelList.size()>0) resultList.add(levelList); } return resultList; }到此这篇关于java二叉树的几种遍历递归与非递归实现代码的文章就介绍到这了,更多相关java二叉树内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
在前面一文,说过二叉树的递归遍历算法(二叉树先根(先序)遍历的改进),此文主要讲二叉树的非递归算法,采用栈结构总结先根遍历得到的非递归算法思想如下:1)入栈,主
用java实现的数组创建二叉树以及递归先序遍历,递归中序遍历,递归后序遍历,非递归前序遍历,非递归中序遍历,非递归后序遍历,深度优先遍历,广度优先遍历8种遍历方
本文实例讲述了C++非递归队列实现二叉树的广度优先遍历。分享给大家供大家参考。具体如下:广度优先非递归二叉树遍历(或者说层次遍历):voidwidthFirst
C++数据结构二叉树(前序/中序/后序递归、非递归遍历)二叉树的性质:二叉树是一棵特殊的树,二叉树每个节点最多有两个孩子结点,分别称为左孩子和右孩子。例:实例代
利用二叉链表存储,并且利用递归的方法实现二叉树的遍历(前序遍历、中序遍历和后续遍历)操作。c语言具体实现代码如下:#include#include#includ