文章目录
链表中环的入口结点问题
一、题目描述
输出示例:
二、题目解析
1、解题思路
解题思路分为两部分:
- 遇到链表中环的问题优先考虑双指针里的快慢指针,快指针就是一次走两个路径,慢指针则只走一个路径,只要快慢指针相遇就返回该结点位置。只要链表中存在环,那么快慢指针必定会相遇。
- 快指针从头开始,慢指针从相遇点开始,二者同时开始走,再次相遇时的位置必定是环的入口处。
2、证明结论
- 必定会相遇
设置快慢指针
fast
和low
,fast每次走两步,low每次走一步。假如有环,两者一定会相遇(因为low一旦进环,可看作fast在后面追赶low的过程,每次两者都接近一步,最后一定能追上)。
- 必定是环的入口处
链表头到环入口长度 ——
a
环入口到相遇点长度为——b
相遇点到环入口长度为——c
当fast与slow相遇时,fast走过的距离为a + k(b + c) + b,而slow走过的距离为a + b,因为fast是slow速度的两倍,则有a+k(b + c)+b = 2*(a+b),化简得出a=b(k-1)+kc
;此时slow节点在相遇点,由表达式可以看出当k为一时,a=c,那么我们让快指针从起点开始走,刚好可以让快慢指针在环入口处相遇。(k是fast走过的环数)
三、代码实现与解释
1、题目源码
/\*
struct ListNode {
int val;
struct ListNode \*next;
ListNode(int x) :
val(x), next(NULL) {
}
};
\*/
class Solution {
public:
ListNode\* EntryNodeOfLoop(ListNode\* pHead) {
ListNode\* slow=flpointer(pHead);
if(slow==NULL)
return NULL;
ListNode\* fast=pHead;
while(fast!=slow){
fast=fast->next;
slow=slow->next;
}
return slow;
}
ListNode\* flpointer(ListNode\* head){
if(head==NULL)
return NULL;
ListNode\* fast=head;
ListNode\* slow=head;
while(fast!=NULL&&fast->next!=NULL){
fast=fast->next->next;
slow=slow->next;
if(slow==fast)
return slow;
}
![img](https://img-blog.csdnimg.cn/img_convert/8827b828a5356b7a64067d1fd23bc15e.png)
![img](https://img-blog.csdnimg.cn/img_convert/b1e4c45c7fa9c974c04aaf3271aee96d.png)
**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**
**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**
**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**
618545628)**
**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**