【力扣】203、环形链表 II

142. 环形链表 II

要解决这道题,首先需要对问题进行拆解:

  1. 确定链表是否存在环
  2. 确定环的入口点

如何判断是否存在环呢?这个比较容易想到,使用快慢指针即可判断链表是否存在环。我们定义两个指针:

ListNode slow = head;
ListNode fast = head;

让 fast 指针的移动速度是 slow 指针的两倍即可,当它们再次相遇时,说明 fast 指针比 slow 指针多走了一圈,并重新追上 slow 指针了,此时可以说明链表存在环。

while(fast != null && fast.next != null) {
    slow = slow.next;
    fast = fast.next.next;
    // 如果慢指针追上快指针,说明存在环
    if(slow == fast) {
        ...
    }
}
return null;

如何确定环的入口点呢?这涉及到数学推导,这一步不太容易想到:

让我们假设三个变量 x,y,z

可以得到公式如下:

slow指针走过的距离 * 2 = fast指针走过的距离
于是得到等式如下:
2(x + y) = (x + y) + n(y + z)		// n(y+z)表示fast指针绕环的长度
x + y = n(y + z)
x = nz + (n - 1)y
x = (n - 1)(z + y) + z

因此我们可以知道,在 slow 指针和 fast 指针相遇的节点处,满足该等式:x = (n - 1)(z + y) + z

这个式子表示什么呢?表示一个指针从头节点处出发,到环型入口处经过的距离 x 等于另一个指针从 slow 和 fast 相交的节点处出发,经过 z + (n - 1)(z + y),即走过 z 距离并绕环 n-1 圈,至于这个 n 是多少我们不必知道,于是可以得到以下代码:

// 通过数学规律发现,相交的节点到环的入口处的节点数等于头节点到环入口处的节点数
ListNode temp = head;
// 如果存在环,必定不会死循环
while(temp != slow) {
    temp = temp.next;
    slow = slow.next;
}
return slow;

至此,题解,完整 Java 代码如下:

/**
 * 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) {
        ListNode slow = head;
        ListNode fast = head;
        while(fast != null && fast.next != null) {
            slow = slow.next;
            fast = fast.next.next;
            // 如果慢指针追上快指针,说明存在环
            if(slow == fast) {
                // 通过数学规律发现,相交的节点到环的入口处的节点数等于头节点到环入口处的节点数
                ListNode temp = head;
                // 如果存在环,必定不会死循环
                while(temp != slow) {
                    temp = temp.next;
                    slow = slow.next;
                }
                return slow;
            }
        }
        return null;
    }
}
相关推荐
shymoy3 分钟前
Radix Sorts
数据结构·算法·排序算法
木向1 小时前
leetcode92:反转链表||
数据结构·c++·算法·leetcode·链表
阿阿越1 小时前
算法每日练 -- 双指针篇(持续更新中)
数据结构·c++·算法
小爬虫程序猿4 小时前
如何利用Python解析API返回的数据结构?
数据结构·数据库·python
pianmian18 小时前
python数据结构基础(7)
数据结构·算法
ChoSeitaku11 小时前
链表交集相关算法题|AB链表公共元素生成链表C|AB链表交集存放于A|连续子序列|相交链表求交点位置(C)
数据结构·考研·链表
偷心编程11 小时前
双向链表专题
数据结构
香菜大丸11 小时前
链表的归并排序
数据结构·算法·链表
jrrz082811 小时前
LeetCode 热题100(七)【链表】(1)
数据结构·c++·算法·leetcode·链表
@小博的博客12 小时前
C++初阶学习第十弹——深入讲解vector的迭代器失效
数据结构·c++·学习