Java?Morris遍历算法及在二叉树中应用的方法是什么


这篇文章主要介绍了JavaMorris遍历算法及在二叉树中应用的方法是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaMorris遍历算法及在二叉树中应用的方法是什么文章都会有所收获,下面我们一起来看看吧。

一.Morris遍历

1.什么是Morris遍历

Morris遍历是一种用于二叉树遍历的算法,它可以在不使用栈或队列的情况下实现中序遍历。该算法的时间复杂度为O(n),空间复杂度为O(1)。

2.基本思想

Morris遍历的基本思想是,利用叶子节点的空指针来存储临时信息,以达到节省空间的目的。具体来说,对于当前遍历到的节点,如果它有左子节点,就找到左子树中最右边的节点,将其右子节点指向当前节点,然后将当前节点更新为其左子节点。如果它没有左子节点,就输出当前节点的值,并将当前节点更新为其右子节点。重复以上步骤,直到遍历完整棵树。

3.Morris遍历的优点和缺点

Morris遍历的优点是空间复杂度低O(1),但它的缺点是会改变原来的二叉树结构,因此需要在遍历完后还原二叉树。此外,该算法可能会比递归或使用栈的算法稍微慢一些。

4.二叉树的线索化

二叉树的线索化,主要是利用了叶子结点中的空指针域,存放了在某种遍历顺序下的前驱或者后续结点,从而达到了线索二叉树的目的

例如,下图中序遍历结果展示如下,根据中序遍历对空指针域进行线索化

线索化的二叉树为下, 画在左边表示left结点指向,画在右边表示right指向,例如7的前驱结点为5,那么7的left指向5,7的后继节点为1,那么7的right指向1

由此,我们可能总结出这样的结论:中序遍历二叉树的指向当前结点的结点为当前结点的左子树的最右端结点,例如指向1的结点为1的左节点2的最右端结点为7,指向2结点的为2的左节点4的最右端结点4.这一点在之后的morris遍历中很重要

具体的代码实现如下:

publicclassInorderThreadedBinaryTree{privateThreadTreeNodepre=null;publicvoidthreadedNodes(ThreadTreeNodenode){//如果node==null,不能线索化if(node==null){return;}//1、先线索化左子树threadedNodes(node.left);//2、线索化当前结点//处理当前结点的前驱结点//以8为例来理解//8结点的.left=null,8结点的.leftType=1if(node.left==null){//让当前结点的左指针指向前驱结点node.left=pre;//修改当前结点的左指针的类型,指向前驱结点node.leftType=1;}//处理后继结点if(pre!=null&&pre.right==null){//让当前结点的右指针指向当前结点pre.right=node;//修改当前结点的右指针的类型=pre.rightType=1;}//每处理一个结点后,让当前结点是下一个结点的前驱结点pre=node;//3、线索化右子树threadedNodes(node.right);}}classThreadTreeNode{intval;ThreadTreeNodeleft;//0为非线索化,1为线索化intleftType;ThreadTreeNoderight;//0为非线索化,1为线索化intrightType;publicThreadTreeNode(intval){this.val=val;}}

但是在实现Morris遍历的时候,并不需要把结点的左节点线索化,只需要把结点的右节点进行线索化即可,具体的原因在下面进行分析.

二.中序Morris遍历

1.中序Morris遍历的分析

上面我们说了Morris遍历的时候只需要线索化右节点,这里给大家进行解释.当我们在中序遍历一棵树的时候,还比如是这样一棵树,我们一步步的node.left来到了6这个结点,这个结点的left为空,所以我们打印6这个结点的值,此时我们需要返回上一个结点,如果我们是要中序递归进行遍历的话,需要返回上一个栈,而我们Morris遍历的时候无法进行递归的返回,所以这个时候我们只需要把6的right结点进行线索化,这个时候6的right指向4,我们就可以返回到4,把4这个结点进行打印,4也线索化返回了2,把2进行打印,然后进行2的right结点5,5的left结点为空,因此打印5,之后进入到5的right结点7,打印7,7的right结点也进行了线索化,进入7的right结点为1,然后打印1,进入3结点并且打印

因为最好不要改变树的结构,所以我们在打印的时候,将线索化的结点的right结点置为空.

2.中序Morris遍历的思路

Morris遍历是利用了线索二叉树的思想,在遍历的过程中不适用栈,从而达到了空间复杂度为O(1)

具体的实现如下:

1.初始化当前的结点为根结点

2.若当前的结点的左节点为空,则输出当前结点,然后遍历当前结点的右子树,即'curr=curr.right'

3.若当前结点的左节点不为空,则找到当前结点的前驱节点,即当前结点左节点的最右侧结点,记为'prev'

  • 如果'prev.right''为空,则将pre.right指向curr结点,然后遍历当前结点的左子树,即'curr=curr.left'

  • 如果'prev.right''不为空,说明已经遍历完了当前节点的左子树,断开 `prev.right` 的连接,即'prev.left=null',输出当前节点,然后遍历当前节点的右子树,即 `curr=curr.right`.

3.具体的代码实现

publicclassMorris{/***将当前根结点中序遍历的结果存储到list集合中*@paramroot根结点*@return中序遍历的结合*/publicList<Integer>inorderTraversal(TreeNoderoot){List<Integer>res=newArrayList<>();TreeNodecurr=root;while(curr!=null){if(curr.left==null){//左子树为空,则输出当前节点,然后遍历右子树res.add(curr.val);//如果要求直接打印,直接输出System.out.println(curr.val);curr=curr.right;}else{//找到当前节点的前驱节点TreeNodeprev=curr.left;while(prev.right!=null&&prev.right!=curr){prev=prev.right;}if(prev.right==null){//将前驱节点的右子树连接到当前节点prev.right=curr;curr=curr.left;}else{//前驱节点的右子树已经连接到当前节点,断开连接,输出当前节点,然后遍历右子树prev.right=null;res.add(curr.val);//如果要求直接打印,直接输出System.out.println(curr.val);curr=curr.right;}}}returnres;}}classTreeNode{intval;TreeNodeleft;TreeNoderight;TreeNode(intx){val=x;}}

测试:

还是这样一颗二叉树,输出如下:

[6, 4, 2, 5, 7, 1, 3]

三.前序Morris遍历

1.前序Morris遍历的思路

前序和中序的遍历很想,只不过在打印(收集结点信息的时候不同),中序遍历是在当前结点的左节点为空(curr.left==null),或者当前结点已经被线索化(prev.right==curr)的时候进行打印,仔细观察前序遍历的过程,我们通过修改打印的顺序即可.前序遍历是在当前结点的左节点为空(curr.left==null),或者当前结点没有被线索化(prev.right==null)的时候进行打印

具体的思路如下:

1.初始化当前的结点为根结点

2.若当前的结点的左节点为空,则输出当前结点,然后遍历当前结点的右子树,即'curr=curr.right'

3.若当前结点的左节点不为空,则找到当前结点的前驱节点,即当前结点左节点的最右侧结点,记为'prev'

  • 如果'prev.right''为空,输出当前节点,然后将pre.right指向curr结点,然后遍历当前结点的左子树,即'curr=curr.left'

  • 如果'prev.right''不为空,说明已经遍历完了当前节点的左子树,断开 `prev.right` 的连接,即'prev.left=null',然后遍历当前节点的右子树,即 `curr=curr.right`.

2.具体的代码实现

publicList<Integer>preorderTraversal(TreeNoderoot){List<Integer>res=newArrayList<>();TreeNodecurr=root;while(curr!=null){if(curr.left==null){//左子树为空,则输出当前节点,然后遍历右子树res.add(curr.val);//如果要求直接打印,直接输出System.out.println(curr.val);curr=curr.right;}else{//找到当前节点的前驱节点TreeNodeprev=curr.left;while(prev.right!=null&&prev.right!=curr){prev=prev.right;}if(prev.right==null){res.add(curr.val);//如果要求直接打印,直接输出System.out.println(curr.val);//将前驱节点的右子树连接到当前节点prev.right=curr;curr=curr.left;}else{//前驱节点的右子树已经连接到当前节点,断开连接,输出当前节点,然后遍历右子树prev.right=null;curr=curr.right;}}}returnres;}

测试:

publicstaticvoidmain(String[]args){TreeNoderoot=newTreeNode(1);root.left=newTreeNode(2);root.left.right=newTreeNode(5);root.left.right.right=newTreeNode(7);root.right=newTreeNode(3);root.left.left=newTreeNode(4);root.left.left.left=newTreeNode(6);System.out.println(preorderTraversal(root));}

还是这样一颗二叉树,输出如下:

[1, 2, 4, 6, 5, 7, 3]

四.后序Morris遍历

1.后序Morris遍历的思路

后序Morris遍历实现起来有一定的难度,但是基本代码还是不变,只是在打印的地方有略微的区别,

具体的思路如下:

1.初始化当前的结点为根结点

2.若当前的结点的左节点为空,则输出当前结点,然后遍历当前结点的右子树,即'curr=curr.right'

3.若当前结点的左节点不为空,则找到当前结点的前驱节点,即当前结点左节点的最右侧结点,记为'prev'

  • 如果'prev.right''为空,然后将pre.right指向curr结点,然后遍历当前结点的左子树,即'curr=curr.left'

  • 如果'prev.right''不为空,此时进行逆序存储,说明已经遍历完了当前节点的左子树,断开 `prev.right` 的连接,即'prev.left=null',然后遍历当前节点的右子树,即 `curr=curr.right`.

2.具体的代码实现

publicList<Integer>postorderTraversal(TreeNoderoot){List<Integer>res=newArrayList<>();TreeNodedump=newTreeNode(0);//建立一个临时结点dump.left=root;//设置dump的左节点为rootTreeNodecurr=dump;//当前节点为dumpwhile(curr!=null){if(curr.left==null){//左子树为空,则输出当前节点,然后遍历右子树curr=curr.right;}else{//找到当前节点的前驱节点TreeNodeprev=curr.left;while(prev.right!=null&&prev.right!=curr){prev=prev.right;}if(prev.right==null){//将前驱节点的右子树连接到当前节点prev.right=curr;curr=curr.left;}else{reverseAddNodes(curr.left,prev,res);//前驱节点的右子树已经连接到当前节点,断开连接,输出当前节点,然后遍历右子树prev.right=null;curr=curr.right;}}}returnres;}privatevoidreverseAddNodes(TreeNodebegin,TreeNodeend,List<Integer>res){reverseNodes(begin,end);//将begin到end的进行逆序连接TreeNodecurr=end;while(true){//将逆序连接后端begin到end添加res.add(curr.val);if(curr==begin)break;curr=curr.right;}reverseNodes(end,begin);//恢复之前的连接状态}/***将begin到end的进行逆序连接**@parambegin*@paramend*/privatevoidreverseNodes(TreeNodebegin,TreeNodeend){TreeNodeprev=begin;TreeNodecurr=prev.right;TreeNodepost;while(prev!=end){post=curr.right;curr.right=prev;prev=curr;curr=post;}}

测试:

publicstaticvoidmain(String[]args){TreeNoderoot=newTreeNode(1);root.left=newTreeNode(2);root.left.right=newTreeNode(5);root.left.right.right=newTreeNode(7);root.right=newTreeNode(3);root.left.left=newTreeNode(4);root.left.left.left=newTreeNode(6);System.out.println(postorderTraversal(root));}

还是这样一颗二叉树,输出如下:

[6, 4, 7, 5, 2, 3, 1]

关于“JavaMorris遍历算法及在二叉树中应用的方法是什么”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“JavaMorris遍历算法及在二叉树中应用的方法是什么”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注主机评测网行业资讯频道。


上一篇:JavaScript使用splice方法删除数组元素导致的问题怎么解决

下一篇:Vue3中的readonly特性及函数使用方法是什么


Copyright © 2002-2019 测速网 www.inhv.cn 皖ICP备2023010105号
测速城市 测速地区 测速街道 网速测试城市 网速测试地区 网速测试街道
温馨提示:部分文章图片数据来源与网络,仅供参考!版权归原作者所有,如有侵权请联系删除!

热门搜索 城市网站建设 地区网站制作 街道网页设计 大写数字 热点城市 热点地区 热点街道 热点时间 房贷计算器