2023-06-13  阅读(6)
原文作者:Java 极客 原文地址:https://www.javajike.com/article/2425.html

引言

上一篇文章《精通二叉树的“独门忍术”——线索二叉树(上)》提到了线索二叉树的改良,并给出了改良后的“中序遍历”“前序遍历”线索二叉树的定义。本文就来谈谈改良后的“前序遍历”的线索二叉树的转换与遍历算法。

非递归型算法

既然《精通二叉树的“独门忍术”——线索二叉树(上)》中已经给出了“中序遍历”的线索二叉树的转换与遍历算法,那么朴素的想法就是:将“前序遍历”线索二叉树与“中序遍历”线索二叉树进行对比,基于后者来推导出前者的算法。

我们先来对比一下“中序遍历”线索二叉树与“前序遍历”线索二叉树的图示:

202306132119286961.png

202306132119294712.png

图1 “中序遍历”的线索二叉树

202306132119299883.png

202306132119306204.png

图2 “前序遍历”的线索二叉树

对比图2与图1可以看出:

“中序遍历”线索二叉树与“前序遍历”线索二叉树的区别仅仅在于后继节点的位置——前者是当前节点,后者是当前节点的直接右孩子。

因此,我们可以完全照搬“中序遍历”线索二叉树的算法,仅仅将后继节点的代码改一下即可:

202306132119311155.png

202306132119319296.png

递归型算法

202306132119324977.png

202306132119332008.png

202306132119338059.png

2023061321193437410.png

还有别的方法吗?

我们来看看是否可以利用传统线索二叉树——即“中序遍历”的线索二叉树,来实现这一目标:非递归地、不用堆栈来做“前序遍历”。

前序遍历的规则简单归纳就是:递归执行“根”->“左”->“右”。

下面的几张图表示了从树根开始“前序遍历”一部分左子树的过程。其中current指针表示当前位置,蓝色闪电表示该位置进行遍历输出,橙黄色箭头表示current指针移动方向。

2023061321193511011.png

2023061321193576912.png

先将当前节点的前驱节点找到,链接起来便形成“中序遍历”的线索二叉树;同时,当前节点是当前局部线索二叉树的树根,根据“根”->“左”->“右”的前序遍历规则,应该输出当前位置作为“根”信息。

2023061321193640313.png

2023061321193725714.png

将当前节点位置指针向左孩子移动。

2023061321193777715.png

2023061321193843316.png

2023061321193952317.png

2023061321194183418.png

2023061321194379419.png

2023061321194494420.png

2023061321194616721.png

2023061321194784622.png

当前位置指针移动到叶子节点时(这种场景的“特征识别码”是:其左孩子指针指向空节点),输出当前位置之后,向当前节点的右孩子指针方向移动。

2023061321194889523.png

2023061321195098924.png

2023061321195283425.png

2023061321195430026.png

一边移动,一边将之前添加的“前驱->后继”的线索去掉,以便还原成原始二叉树。

这种场景的“特征识别码”是:当前节点是前驱节点的右孩子。

2023061321195542827.png

2023061321195672828.png

根据上面的分析,很容易翻译成如下的基于“中序遍历”的线索二叉树、非递归型、不用堆栈、并且遍历完后还可以恢复成原始二叉树的“神算法”:

2023061321195739429.png

2023061321195830230.png

后记

研究算法,和研究数学问题一样,“一题多解”可以极大拓宽思路和增强想象力、“防止老年痴呆”哦:)

老规矩,留个思考题、答案在下一篇文章揭晓:

最后的这个“神算法”的时间复杂度是多少呢?


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文