传送门:
141. 环形链表 - 力扣(LeetCode)https://leetcode.cn/problems/linked-list-cycle/description/
一.题目描述:
给你一个链表的头节点
head
,判断链表中是否有环。如果链表中有某个节点,可以通过连续跟踪
next
指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数pos
来表示链表尾连接到链表中的位置(索引从 0 开始)。注意:pos
不作为参数进行传递 。仅仅是为了标识链表的实际情况。如果链表中存在环 ,则返回
true
。 否则,返回false
。
二.代码与分析
(1)题目分析
读完题目,其实不难想出解决方案:我们可以定义两个指针,一个快指针,一个慢指针,常规操作是快指针一次走两步,慢指针一次走一步,当如果存在回路,那么经过有限长的时间,快指针一定能追上慢指针,此时返回true;如果循环退出,则说明没有环,返回false。
(2)代码实现
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* struct ListNode *next;
* };
*/
bool hasCycle(struct ListNode *head) {
struct ListNode* slow=head,*fast=head;
while(fast && fast->next)
{
slow=slow->next;
fast=fast->next->next;
if(slow==fast)
return true;
}
return false;
}
三.拓展延申
- 为什么快指针每次走两步,慢指针每次走一步可以?
假设链表带环,两个指针最后都会进入环,快指针先进环,慢指针后进环。当慢指针刚进环时,可能就和快指针相遇了,最差情况下两个指针之间的距离刚好就是环的长度。此时,两个指针每移动一次,之间的距离就缩小一步,不会出现每次刚好是套圈的情况,因此:在满指针走到一圈之前,快指针肯定是可以追上慢指针的,即相遇。
2.快指针一次走3步,4步,...n步可以吗?
假设:快指针每次走3步,慢指针每次走1步,此时快指针肯定先进环,慢指针后来才进环。假设慢指针进环时候,快指针的位置如图所示:
此时按照上述方法来绕环移动,每次快指针走3步,慢指针走1步,是永远不会相遇的,快指针刚好将慢指针套圈了,因此不行。
只有快指针走2步,慢指针走一步才可以,因为环的最小长度是1,即使套圈了两个也在相同的位置。