<强>前言强>
三种遍历的递归写法都很好写,所以总结一下非递归写法。
先贴一张图复习一下三种遍历方式就进入正文啦~
【注:本文所有代码实现中树的结点定义如下:
公共类节点{ int val; 节点离开; 节点对的; 节点的父; 节点(){} 节点(int val) { 这一点。val=val; } }
<强> 1。前序遍历强>
<>强实现思路:强>
前序遍历的顺序是:根结点→左孩子→右孩子
借助一个栈结构先将根结点压入栈,然后循环每次取出栈顶元素,直到栈为空。如果当前结点有右孩子就先将右孩子压入栈中,如果当前结点有左孩子就将左孩子压入栈中,这里注意顺序,因为栈的结构是先进后出的,为了保证先遍历到左孩子就应该后压左孩子~
<强>代码实现:强>
【代码使用直接输出的方式打印中序遍历结果,也可以放入一个列表中存储~】
公共空间预订(节点){ System.out.println(“预购:”); 如果(头!=null) { Stack堆栈=new Stack<在(); stack.push(头); 而(! stack.isEmpty ()) { 头=stack.pop (); System.out.print(头。val + " "); 如果(头。没错!=null) stack.push (head.right); 如果(头。离开!=null) stack.push (head.left); } } }
<强> 2,中序遍历强>
<>强实现思路:强>
中序遍历的顺序:左孩子→根结点→右孩子
借助栈结构,将当前结点的左孩子依次入栈直到没有左孩子了就弹出栈顶元素并打印,如果弹出的结点有右孩子就将右孩子的左边界依次入栈,循环…
比如文章开始的那个图中,先将A, B, D依次入栈,此时栈顶元素是D,弹出并打印,D结点有右孩子,将D的右孩子的左边界依次入栈,压入结点G,结点G没有左孩子所以弹出并打印G,此时栈顶元素是B,循环…直到栈中为空且当前结点为空时遍历结束。
<强>代码实现:强>
公共静态孔隙inOrderTraverse(节点){ System.out.println(“顺序:”); 如果(头!=null) { Stack堆栈=new Stack<在(); 而(! stack.isEmpty() | |头!=null) { 如果(头!=null) {//当前节点不为空,将自己压进栈并将自己的左孩子作为当前节点(压入左边界) stack.push(头); 头=head.left; 其他}{//当前节点为空(没有左孩子了),将栈顶元素弹出作为当前节点,并将当前节点的右孩子压进栈 头=stack.pop (); System.out.print(头。val + " "); 头=head.right; } } } }
<强> 3。后序遍历强>
<>强实现思路:强>
后序遍历的顺序:左孩子→右孩子→根结点
仔细想一下,打印每个结点需要访问根结点三次,先从根结点开始找到左孩子,返回再找到右孩子,再返回到根结点,需要访问根结点三次,直接按照当前顺序进行遍历不知如何下手,那么我们可以换一个角度去考虑。
栈结构是先进后出的,那我们不妨借助一个栈先存储根结点→右孩子→左孩子的结果,再将其依次弹出就是后序遍历的顺序了。
那实现根结点→右孩子→左孩子的方法就很简单了,回忆一下刚才说的前序遍历的方式,前序遍历是先要左孩子,就后压入左孩子,反之,将前序遍历的逻辑改写为:弹出每个栈顶结点作为当前结点并存储到一个辅助栈中,如果当前结点有左孩子就先压入左孩子,如果有右孩子就后压入右孩子,每次取栈顶弹出到辅助栈中。最后将得到的辅助栈中元素依次弹出得到的就是后序遍历的结果。
<强>代码实现:强>
公共静态孔隙posOrderTraverse(节点){ System.out.println (“pos-order”); 如果(头!=null) { Stack,stack1=new Stack<的在(); Stack ,stack2=new Stack<的在();//辅助栈,存储根→右→左的结果 stack1.push(头); 而(! stack1.isEmpty ()) { 头=stack1.pop (); stack2.push(头);//有左孩子就先压入左孩子 如果(头。离开!=null) stack1.push (head.left);//有右孩子就后压入右孩子 如果(头。没错!=null) stack1.push (head.right); }//逆序打印根→右→左的结果,就是后序遍历的结果 而(! stack2.isEmpty ()) System.out.print (stack2.pop ()。val + " "); } } Java语言实现非递归实现树的前中后序遍历总结