1.题目

2.思路

方法一——哈希表记录节点

思路很简单,记录一下每个节点出现的次数,如果某个节点出现了两次,代表此时有环,并且是环的入口,直接返回即可。

时间复杂度O(N)

空间复杂度O(N)

public class Solution {public ListNode detectCycle(ListNode head) {ListNode pos = head;Set<ListNode> visited = new HashSet<ListNode>();while (pos != null) {if (visited.contains(pos)) {return pos;} else {visited.add(pos);}pos = pos.next;}return null;}
}

方法二——快慢指针找到环 + temp指针找到环的入口

想法:首先把问题拆分成两个问题,第一个问题判断是否有环。第二个问题,如果有环的话,如何判断环的入口呢?并且保证空间复杂度为O(1)。

第一个问题很好解决。直接快慢指针就可以。

slow指针走一步

fast指针走两步

如果链表存在环,那么必然fast指针和slow指针会相遇。为什么呢?

简单的正面如下:

假如slow刚好走到环的入口,此时fast指针必然在环中的任意一个位置,此时他们之间的距离为length, 这个长度length 必然小于环的大小。

每一次,fast 都会比slow多走一步,所以下次他们之间的距离为length-1.

再下一次他们的距离为length - 2.

...

直到他们两相遇!!!

并且可以得到的是,slow不会走过一圈,他们就会相遇!因为这个距离length小于环的大小。

如果链表存在环,那么如何找到环的入口呢?(变形:如何求出环的大小?)

先说结论。此时定义一个temp指针,指向 head, 让slow 和 temp同时移动,他们相等的时候就是环的入口。

证明如下:(借用官方题解的一个图)

a : 起点到环的入口的举例为a

b:环的入口到快慢指针相遇的位置。

c:环的大小减去b的距离

首先可以得知slow指针移动的距离: a + b

然后可以得知fast指针移动的距离:a + b + n * (b + c)

然后利用fast指针速度是slow指针的两倍,所以在相同的时间下,距离也应该是两倍关系。

所以有2 *(a + b) = a + b + n * (b + c)

因为我们要计算a,所以我们想办法把a放在左边,那么化简可以得到

a = nb - b + nc

这样看不出来什么东西,所以进一步想,因为有环的大小为b + c,所以想办法凑在一起。有

a = (n - 1) * (b + c) + c

这时候可以发现,

左边的值等于一个temp指针从起点走到环的入口

右边的值相当于是slow指针先走距离c, 到达环的入口,然后再走n - 1 圈,直到两个相遇!!!!!!!

所以有结论:当快慢指针相等的时候,让slow 和 temp同时移动,他们相等的时候就是环的入口。

时间复杂度O(N)

空间复杂度O(1)

/*** Definition for singly-linked list.* class ListNode {*     int val;*     ListNode next;*     ListNode(int x) {*         val = x;*         next = null;*     }* }*/
public class Solution {public ListNode detectCycle(ListNode head) {// 时间O(n)// 空间O(1),这里使用的是三个指针,不占用空间ListNode slow = new ListNode(-1), fast = new ListNode(-1);slow = head;fast = head;while(fast != null){if(fast.next != null && fast.next.next != null){fast = fast.next.next;}elsereturn null;slow = slow.next;if(slow == fast){ //代表有循环// 找到环的入口ListNode temp = head;while(temp != slow){temp = temp.next;slow = slow.next;}return temp;}}return null;}
}

3.结果

4.类似的题

判断是否有环?

判断环的大小?

删除倒数第K个数?

面试题 02.08. 环路检测-快慢指针+如何找到环的入口?(证明)Java相关推荐

  1. 程序员面试金典 - 面试题 02.08. 环路检测(快慢指针)

    1. 题目 给定一个有环链表,实现一个算法返回环路的开头节点. 有环链表的定义:在链表中某个节点的next元素指向在它前面出现过的节点,则表明该链表存在环路. 示例 1: 输入:head = [3,2 ...

  2. leetcode面试题 02.08. 环路检测

    给定一个有环链表,实现一个算法返回环路的开头节点. 有环链表的定义:在链表中某个节点的next元素指向在它前面出现过的节点,则表明该链表存在环路. 示例 1: 输入:head = [3,2,0,-4] ...

  3. 证明利用快慢指针寻找有环单链表中环的起点算法

    问题:给定一个有环单链表,找到链表中环的起点,也就是说,找到下图中的单链表中Join点: (本图来源于http://www.cnblogs.com/xudong-bupt/p/3667729.html ...

  4. 证明-快慢指针找链表环

    快慢指针证明 (1)先证明两个指针可以相遇 1.如果链表没有环,那么快指针比慢指针先到达尾部(指向null). 2.如果链表有环的话,因为快指针走的比慢指针快,所以在环中相遇的过程可以看作是快指针从环 ...

  5. 试编写一个将双向循环链表逆置的算法_图解:链表的快慢指针,解决 80% 的链表面试题!...

    一.前言 链表是基本的数据结构之一,它与数组不同,数组在内存中存储,需要一块连续的内容空间来存储,对内存的要求比较高.例如我们需要 100MB 大小的数组,内存中就必须有一段连续的 100MB 的内存 ...

  6. 剑指offer-面试题23:链表中环的入口节点 快慢指针+双指针

    题目描述 给一个链表,若其中包含环,请找出该链表的环的入口结点,否则,输出null.如图所示,节点3是链表中环的入口节点 本题的解决分为两步:(1)确定链表中是否有环(2)如果有环,确定环的入口节点. ...

  7. LeetCode 例题精讲 | 05 双指针×链表问题:快慢指针

    点击关注上方"五分钟学算法", 设为"置顶或星标",第一时间送达干货. 转自面向大象编程 本期例题: LeetCode 876 - Middle of the ...

  8. 一文学会链表快慢指针解题技巧

    前言 上文 我们详细地学习了链表的基本概念,优缺点,也带大家一步步由浅入深地学习了链表的翻转技巧,这一篇我们来看看链表的另一个解题技巧:快慢指针. 快慢指针在面试中出现的概率也很大,也是务必要掌握的一 ...

  9. 动画:链表快慢指针解题技巧

    点击蓝色"五分钟学算法"关注我哟 加个"星标",天天中午 12:15,一起学算法 作者 | 码海 前言 这一篇我们来看看链表的另一个解题技巧:快慢指针. 快慢指 ...

最新文章

  1. Oracle根据日期区间查询Date类型的数据
  2. 让Transformer的推理速度提高4.5倍,这个小trick还能给你省十几万
  3. ShakeDrop:深度残差学习中的 ShakeDrop 正则化
  4. va_list和va_start和((A*)0)-a
  5. 探究:如何判断Delphi中的对象指针是否可用
  6. Lua 文件 I/O
  7. Eclipse CDT Hello World工程makefile分析
  8. Qt QByteArray 转换 QString QT5.12.7
  9. 循环次数几次_圆柱模板循环使用次数是多少呢
  10. win7更改用户登录或注销方式的操作教程
  11. bzoj1024 [SCOI2009]生日快乐 结论+dfs
  12. 围观京东云,您有一份区块链技术礼包待查收!
  13. android 7.0 截图,Android,_7.0系统拍照后,使用系统截图功能,截图保存时崩溃如何解决,Android - phpStudy...
  14. nginx 编译安装,服务查看启停,以及模块的追加编译
  15. 家谱处理 (30 分)(map映射)
  16. 大话设计模式 设计模式书籍 编程书籍
  17. 修改Android系统配置文件build.prop
  18. GIS应用技巧之密度分析
  19. H2数据库入门,看这篇就对了
  20. 今天,爱思唯尔发布2022“中国高被引学者” 榜单

热门文章

  1. 高一计算机课期中考试总结反思,2017高一数学期中考试反思总结
  2. Pentaho Data Integration(Kettle) 6.0
  3. 图解IFRS 9 金融工具(13)实施时间及结语
  4. 手机黑名单,拦截电话和短信,清除通话记录
  5. POJ 3669(优先队列BFS)(对地图进行优化)
  6. ThinkPad开机停留在boot menu界面、进不了系统的解决方法
  7. javaScript字符串比较
  8. 计算机组成原理——总线连接方式
  9. Qt 做个简易的计算器,加减乘除平方开根
  10. MSDN for VS2005