[笔试题]判断链表是否有环及环的位置问题

2024-02-27 19:08

本文主要是介绍[笔试题]判断链表是否有环及环的位置问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

首先,这个问题的最简单情况就是判断两个链表是否相交问题。我写过一篇文章,见这里:http://hi.baidu.com/microgrape/blog/item/5575771e6d3f9c03304e15d8.html

===============================================================

然后引申到如何判断一个链表存在环的问题。

关于这个问题,解决方案及解释如下:

用两个指针来遍历这个单向链表,第一个指针p1,每次走一步;第二个指针p2,每次走两步;当p2 指针追上 p1的时候,就表明链表当中有环路了。

关于这个解法最形象的比喻就是在操场当中跑步,速度快的会把速度慢的扣圈.

可以证明,p2追赶上p1的时候,p1一定还没有走完一遍环路,p2也不会跨越p1多圈才追上.

我们可以从p2和p1的位置差距来证明,p2一定会赶上p1但是不会跳过p1的.

因为p2每次走2步,而p1走一步,所以他们之间的差距是一步一步的缩小,
4 3 2 1 0  到0的时候就重合了.

根据这个方式,可以证明,p2每次走三步以上,并不总能加快检测的速度,反而有可能判别不出有环.

比如,在环的周长L是偶数的时候,初始p2和p1相差奇数的时候,p2每次走三步,就永远和p1不重合,因为他们之间的差距是:  
5   3    1 ,   L- 1 , L- 3

 

附加几点解释:虽然说p1看来“相对”于p2并不前进,但是“绝对”来说,它还是前进的,那么迟早它会跑到圆圈上,而且不再出来。p2就更是了。所以,尽管有可能p1还没有进圆圈,p2已经跑完几圈了(想象一下圆圈很小,但是前面的直道部分很长的情况),但是p1迟早要进圈,演变成跑步问题。

========================================================

下面则再进一步:如何判断链表的环开始的位置呢?也就是环与链表的交点。

这个问题就更靠近脑筋急转弯的方向了。首先想到,如果还是p1和p2一起跑圈,那仍然会发生很多次相遇,但是这个相遇跟起点并无关系,想求出入口不可能。

那我们就分析一下,p1和p2相遇的时候到底发生了什么。在示意图中,假设p1和p2在P点第一次相遇。从环路入口到P点的距离为x,从P点到环路入口的另一段的距离为y。

那么显然,第一次相遇的时候,p1走过的总长度为x+z。那p2呢?我们首先假设p2比p1只多跑了一圈(方便理解,但并不影响最后结论,下面再证明所有的情况)。那么容易得到,p2走过的总长度为z+x+y+x。这个结论很显然也是表明p2比p1多跑了一圈(一圈的长度就是x+y,与前面的分析吻合)。由于p2的速度是p1的两倍,很容易得到一个结论:z+x+y+x = 2 * (x+z)。也就是z = y。也就是说,相遇的点离环路入口的距离与链表直道的长度一样。

呵呵,得到这个结论似乎就离我们的答案很近了:如何才能找到这个入口位置呢?显然,让一个指针从相遇的P点出发,另外一个指针从链表头出发,以相同的速度前进,那么它们必然会在O点也就是环路入口相遇!

补充证明:设p2跑了n圈之后,才跟p1相遇(但请注意:p1一定是在自己的第一圈,因为不可能发生两个指针都在圈内,p2超过了p1但是没相遇的情况,前面的分析里说了),那么可以得到下面的等式:

z + n * (x+y) + x = 2 * (x+z)

最后可以得到下面的式子:z = n*(x+y) - x = (n-1)*(x+y) + y。 也就是说直道的长度等于n-1圈的长度加上y的长度,仍然不影响上面的结论:如果两个指针速度相同,一个指针从head出发,另一个从P点出发(跑n-1圈,再跑y),所花时间相等。它们一定会在O点相遇!

==========================================================

这篇关于[笔试题]判断链表是否有环及环的位置问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/753330

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

mybatis和mybatis-plus设置值为null不起作用问题及解决

《mybatis和mybatis-plus设置值为null不起作用问题及解决》Mybatis-Plus的FieldStrategy主要用于控制新增、更新和查询时对空值的处理策略,通过配置不同的策略类型... 目录MyBATis-plusFieldStrategy作用FieldStrategy类型每种策略的作

linux下多个硬盘划分到同一挂载点问题

《linux下多个硬盘划分到同一挂载点问题》在Linux系统中,将多个硬盘划分到同一挂载点需要通过逻辑卷管理(LVM)来实现,首先,需要将物理存储设备(如硬盘分区)创建为物理卷,然后,将这些物理卷组成... 目录linux下多个硬盘划分到同一挂载点需要明确的几个概念硬盘插上默认的是非lvm总结Linux下多

Python Jupyter Notebook导包报错问题及解决

《PythonJupyterNotebook导包报错问题及解决》在conda环境中安装包后,JupyterNotebook导入时出现ImportError,可能是由于包版本不对应或版本太高,解决方... 目录问题解决方法重新安装Jupyter NoteBook 更改Kernel总结问题在conda上安装了

pip install jupyterlab失败的原因问题及探索

《pipinstalljupyterlab失败的原因问题及探索》在学习Yolo模型时,尝试安装JupyterLab但遇到错误,错误提示缺少Rust和Cargo编译环境,因为pywinpty包需要它... 目录背景问题解决方案总结背景最近在学习Yolo模型,然后其中要下载jupyter(有点LSVmu像一个

解决jupyterLab打开后出现Config option `template_path`not recognized by `ExporterCollapsibleHeadings`问题

《解决jupyterLab打开后出现Configoption`template_path`notrecognizedby`ExporterCollapsibleHeadings`问题》在Ju... 目录jupyterLab打开后出现“templandroidate_path”相关问题这是 tensorflo

如何解决Pycharm编辑内容时有光标的问题

《如何解决Pycharm编辑内容时有光标的问题》文章介绍了如何在PyCharm中配置VimEmulator插件,包括检查插件是否已安装、下载插件以及安装IdeaVim插件的步骤... 目录Pycharm编辑内容时有光标1.如果Vim Emulator前面有对勾2.www.chinasem.cn如果tools工

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动