如何判断单链表里面是否有环?

本文详细介绍了如何使用双指针技术判断单链表中是否存在环,分析了环的长度及首个环内节点的确定方法。通过数学推导,阐述了不同速度指针相交的条件,并探讨了两个单链表交点的判断技巧。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

如何判断单链表里面是否有环?

算法的思想是设定两个指针p, q,其中p每次向前移动一步,q每次向前移动两步。那么如果单链表存在环,则p和q相遇;否则q将首先遇到null。
这里主要理解一个问题,就是为什么当单链表存在环时,p和q一定会相遇呢?

假定单链表的长度为n,并且该单链表是环状的,那么第i次迭代时,p指向元素i mod n,q指向2i mod n。因此当i≡2i(mod n)时,p与q相遇。而i≡2i(mod n) => (2i - i) mod n = 0 => i mod n = 0 => 当i=n时,p与q相遇。这里一个简单的理解是,p和q同时在操场跑步,其中q的速度是p的两倍,当他们两个同时出发时,p跑一圈到达起点,而q此时也刚 好跑完两圈到达起点。
那么当p与q起点不同呢?假定第i次迭代时p指向元素i mod n,q指向k+2i mod n,其中0<k<n。那么i≡(2i+k)(mod n) => (i+k) mod n = 0 => 当i=n-k时,p与q相遇。

解决方案:

推广:

  1. 如果两个指针的速度不一样,比如p,q,( 0<p<q)二者满足什么样的关系,可以使得两者肯定交与一个节点?

    Sp(i) = pi

    Sq(i) = k + qi

    如果两个要相交于一个节点,则 Sp(i) = Sq(i) => (pi) mod n = ( k+ qi ) mod n =>[ (q -p)i + k ] mod n =0

    => (q-p)i + k = Nn [N 为自然数]

    => i = (Nn -k) /(p-q)

    i取自然数,则当 p,q满足上面等式 即 存在一个自然数N,可以满足Nn -k 是 p - q 的倍数时,保证两者相交。

    特例:如果q 是p 的步长的两倍,都从同一个起点开始,即 q = 2p , k =0, 那么等式变为: Nn=i: 即可以理解为,当第i次迭代时,i是圈的整数倍时,两者都可以交,交点就是为起点。

2.如何判断单链表的环的长度?

这个比较简单,知道q 已经进入到环里,保存该位置。然后由该位置遍历,当再次碰到该q 位置即可,所迭代的次数就是环的长度。

  1. 如何找到链表中第一个在环里的节点?

    假设链表长度是L,前半部分长度为k-1,那么第一个再环里的节点是k,环的长度是 n, 那么当q=2p时, 什么时候第一次相交呢?当q指针走到第k个节点时,q指针已经在环的第 k mod n 的位置。即p和q 相差k个元素,从不同的起点开始,则相交的位置为 n-k, 则有了下面的图:

    从图上可以明显看到,当p从交点的位置(n-k) ,向前遍历k个节点就到到达环的第一个几点,节点k.

算法就很简单: 一个指针从p和q 中的第一次相交的位置起(n-k),另外一个指针从链表头开始遍历,其交点就是链表中第一个在环里的交点。

  1. 如果判断两个单链表有交?第一个交点在哪里?

    这个问题画出图,就可以很容易转化为前面的题目。

    将其中一个链表中的尾节点与头节点联系起来,则很容发现问题转化为问题3,求有环的链表的第一个在环里的节点。

### 创建带头节点的单链表判断是否存在 #### 使用头插法创建带头节点的单链表 为了实现 `CreatList` 函数,可以按照以下逻辑编写代码。此函数通过头插法构建一个带有头节点的单链表。每次读取新数据时,在当前链表头部插入新的节点。 以下是完整的 C++ 实现: ```cpp #include <iostream> using namespace std; // 定义单链表节点结构 struct ListNode { int data; ListNode* next; }; // 使用头插法创建带头节点的单链表 ListNode* CreatList() { ListNode* head = new ListNode(); // 创建头节点 head->next = nullptr; // 初始化为空列表 int value; cout << "Please input the elements of the list (-1 to end):" << endl; while (cin >> value && value != -1) { // 输入直到遇到-1为止 ListNode* newNode = new ListNode(); newNode->data = value; // 设置新节点的数据 // 插入到链表头部 newNode->next = head->next; head->next = newNode; } return head; // 返回带虚拟头节点的链表 } ``` 以上代码实现了使用头插法创建单链表的功能[^3]。注意这里返回了一个带有头节点的链表指针。 --- #### 判断单链表是否存在 要检测单链表是否,可采用 **Floyd 的循查找算法** 或称为 “快慢指针” 方法。其核心思想是利用两个速度不同的指针遍历链表。如果存在,则这两个指针最终会相遇;否则,较快的指针将到达链表末尾。 下面是具体的实现代码: ```cpp bool IsCir(ListNode* head) { if (!head || !head->next) return false; // 如果链表为空或者只有一个节点则无 ListNode* slow = head->next; // 慢指针初始化为第一个实际节点 ListNode* fast = head->next; // 快指针也从第一个实际节点开始 while (fast && fast->next) { slow = slow->next; // 慢指针移动一步 fast = fast->next->next; // 快指针移动两步 if (slow == fast) { // 若两者相遇说明有 return true; } } return false; // 遍历完成未发现 } ``` 这段代码基于 Floyd 循查找算法来判断链表是否存在[^4]。 --- ### 测试程序 下面是一个简单的测试框架,用于验证上述功能是否正常工作: ```cpp int main() { ListNode* head = CreatList(); // 调用头插法创建链表 // 手动制造以便测试 IsCir() /* ListNode* temp = head->next; while (temp->next) { temp = temp->next; } temp->next = head->next; // 形成 */ bool hasCircle = IsCir(head); if (hasCircle) { cout << "The linked list contains a cycle." << endl; } else { cout << "The linked list does not contain a cycle." << endl; } return 0; } ``` 取消注释部分代码即可手动设置以测试 `IsCir()` 函数的行为。 --- ### 总结 上述代码展示了如何使用头插法创建一个带头节点的单链表,并提供了检测链表是否存在的方法。头插法的特点在于它能够快速地在链表前端插入元素,而无需额外操作。然而需要注意的是,这种方法会使输入顺序与存储顺序相反。至于检测方面,“快慢指针” 是一种高效且经典的解决方案。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值