题目描述
leetcode_hot100_141
给你一个链表的头节点 head ,判断链表中是否有环。
如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。注意pos不作为参数进行传递 。仅仅是为了标识链表的实际情况。
如果链表中存在环 ,则返回 true 。 否则,返回 false 。
示例 1:
输入:head = [3,2,0,-4], pos = 1
输出:true
解释:链表中有一个环,其尾部连接到第二个节点。
示例 2:
输入:head = [1,2], pos = 0
输出:true
解释:链表中有一个环,其尾部连接到第一个节点。
示例 3:
输入:head = [1], pos = -1
输出:false
解释:链表中没有环。
提示:
链表中节点的数目范围是 [0, 104]
-105 <= Node.val <= 105
pos 为 -1 或者链表中的一个 有效索引 。
进阶:你能用 O(1)(即,常量)内存解决此问题吗?
正解一
利用哈希表,复杂度为O(n)
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
bool hasCycle(ListNode *head) {
unordered_set<ListNode*> set;
ListNode *p=head;
while(p!=NULL){
if(set.count(p)){//返回p在set中出现的次数
return true;
}
set.insert(p);
p=p->next;
}
return false;
}
};
正解二
思路:快慢指针,一个指针每次走两步,一个指针每次走一步,如果没有环,则快指针最终会指向null,若有环,则快慢指针早晚会相遇。
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
bool hasCycle(ListNode *head) {
if(head==NULL){//率先排除是空链表的情况,不然不存在head->next
return false;
}
ListNode *pf=head->next;
ListNode *ps=head;
while(pf!=NULL && pf!=ps){
if(pf->next==NULL){//每次更新pf前先看看next是否已经到null,万一pf->next==null,则不存在pf->next->next
return false;
}
pf=pf->next->next;
ps=ps->next;
}
if(pf==ps){
return true;
}else{
return false;
}
}
};