【数据结构与算法】(LeetCode)141.环形链表 142.环形链表Ⅱ

【数据结构与算法】(LeetCode)141.环形链表 142.环形链表Ⅱ

文章目录

引言

环形链表问题是数据结构与算法中的经典问题,在面试中出现频率极高。这类问题不仅考察对链表结构的理解,更考验解决问题的思维能力和数学分析能力。本文将详细分析环形链表的判断方法以及环入口节点的定位算法,帮助读者深入理解这一重要问题。

环形链表判断

问题描述

给定一个链表的头节点 head,判断链表中是否存在环。

在这里插入图片描述

解决方案:快慢指针法

快慢指针法是解决环形链表问题的经典方法,其核心思想是使用两个指针以不同速度遍历链表。

bool hasCycle(structListNode*head){structListNode* slow=head,*fast=head;while(fast&&fast->next){ slow=slow->next; fast=fast->next->next;//一定要先让快慢指针走,再判断 因为一开始快慢指针都是headif(slow==fast)return true;}return false;}

原理分析

为什么快慢指针一定能相遇?

假设慢指针进入环时,快指针与慢指针之间的距离为N。由于快指针每次比慢指针多走一步,它们之间的距离会逐次减少:N, N-1, N-2, …, 2, 1, 0。因此最终一定会相遇。

步长选择的数学分析

为什么选择一步和两步?

当快指针每次走三步、四步或更多时,情况会变得更加复杂。以三步为例:

  • 每次移动后,快慢指针之间的距离减少2
  • 当N为偶数时,可以追上
  • 当N为奇数时,会错过,距离变为C-1(C为环长)
在这里插入图片描述

假设slow进环时,fast跟slow的距离是N

fast追击slow时 距离变化为N N-1 N-2 ……2 1 0

所以能追上

在这里插入图片描述

当fast等于3时,每次fast与slow距离就减2

1.当N为偶数时,可以追上

2.当N为奇数时,会错过,这时他们的距离会变为C-1 (假设C为环的长度)

a.如果C-1为偶数(C为奇数),下一轮就追上了

b.如果C-1为奇数(C为偶数),就永远追不上

所以这么一看,追不上的条件是:N为奇数并且C为偶数

在这里插入图片描述

但是这种情况存在吗?

进一步分析表明,当快指针走三步时,追不上的条件是:N为奇数且C为偶数。但通过数学推导可以证明这种情况实际上不可能发生:

假设slow走的距离是L

fast走的距离就是L+x*C+N (x为fast走的圈数)

fast走的距离是slow的3倍

3L=L+x*C+N

化简可得

2L=x*C+N

当C为偶数N为奇数时,x*C为偶数

等号左边是偶数,右边是偶数加奇数(也就是奇数),显然不成立

所以不存在N为奇数,C为偶数的情况

所以fast一定能追上slow

环形链表Ⅱ

题目如下

在这里插入图片描述

方法一

找到fast与slow相遇的位置meet,head与meet同时走,他们相遇位置就是环的入口

证明如下:

在这里插入图片描述

相遇时:

slow走的路程为:L+N

fast走的路程为: L+x*C+N

fast走的是slow的2倍

2(L+N)=L+x*C+N

化简可得

L=(x-1)*C+C-N

所以L的长度就是相遇点转了x-1圈 再走了C-N距离的点

这意味着:从头节点到环入口的距离L,等于从相遇点走(x-1)圈再加(C-N)步。因此,两个指针分别从头节点和相遇点出发,一定会在环入口相遇。

代码如下

structListNode*detectCycle(structListNode*head){structListNode* slow=head,*fast=head;while(fast&&fast->next){ slow=slow->next; fast=fast->next->next;if(slow==fast){structListNode* meet=slow;while(meet!=head){ head=head->next; meet=meet->next;}return meet;}}returnNULL;}

方法二:转换为相交链表问题

算法思路
  1. 找到快慢指针的相遇点
  2. 将环从相遇点处断开
  3. 问题转换为求两个链表的交点问题
在这里插入图片描述

将环形链表断开

newhead=meet->next

meet->next=NULL

这样就转换成了链表相交的问题

structListNode*getIntersectionNode(structListNode*headA,structListNode*headB){structListNode* curA=headA,*curB=headB;int lenA=1,lenB=1;while(curA->next){ curA=curA->next; lenA++;}while(curB->next){ curB=curB->next; lenB++;}int gap=abs(lenA-lenB);//假设法 先假设A长structListNode* longList=headA;structListNode* shortList=headB;if(lenA<lenB){ longList=headB; shortList=headA;}while(gap--){ longList=longList->next;}while(longList){if(longList==shortList)return longList; longList=longList->next; shortList=shortList->next;}returnNULL;}structListNode*detectCycle(structListNode*head){structListNode* slow=head,*fast=head;while(fast&&fast->next){ slow=slow->next; fast=fast->next->next;if(slow==fast){structListNode* meet=slow;structListNode* newhead=meet->next; meet->next=NULL;returngetIntersectionNode(head,newhead);}}returnNULL;}

实际应用与扩展

应用场景

  1. 内存管理:检测内存泄漏或循环引用
  2. 状态机检测:验证有限状态机是否会进入循环状态
  3. 递归检测:判断递归函数是否会无限递归
    meet->next=NULL;
    return getIntersectionNode(head,newhead);
    }}

return NULL;

}

 ## 实际应用与扩展 ### 应用场景 1. **内存管理**:检测内存泄漏或循环引用 2. **状态机检测**:验证有限状态机是否会进入循环状态 3. **递归检测**:判断递归函数是否会无限递归 4. **哈希冲突解决**:在开放定址法中检测哈希表是否已满 

Read more

【C++】第二十六节—C++11(中) | 右值引用和移动语义(续集)+lambda

【C++】第二十六节—C++11(中) | 右值引用和移动语义(续集)+lambda

Hi,我是云边有个稻草人,C++领域博主与你分享专业知识(*^▽^*) 《C++》本篇文章所属专栏—持续更新中—欢迎订阅~ 目录 上节总览,详情见—>【C++】第二十五节—C++11 (上) | 详解列表初始化+右值引用和移动语义 本节总览 (4)右值引用和移动语义在传参中的提效 6. 类型分类 7. 引用折叠 8. 完美转发 四、lambda 1. lambda表达式语法 2. lambda的应用 3. 捕捉列表 4. lambda的原理 接着上节,正文开始—— (4)右值引用和移动语义在传参中的提效 * 查看STL文档我们发现C++11以后容器的push和insert系列的接口否增加的右值引用版本 * 当实参是一个左值时,容器内部继续调用拷贝构造进行拷贝,将对象拷贝到容器空间中的对象 * 当实参是一个右值,容器内部则调用移动构造,右值对象的资源到容器空间的对象上

By Ne0inhk
新手必看!VSCode&PyCharm 配置 OpenCV 超详细教程(支持 Python 和 C++ 双语言)

新手必看!VSCode&PyCharm 配置 OpenCV 超详细教程(支持 Python 和 C++ 双语言)

新手必看!VSCode&PyCharm 配置 OpenCV 超详细教程(支持 Python 和 C++ 双语言) 适用对象:初学者,希望在 VSCode 与 PyCharm 两款常用 IDE 中,学会配置并使用 OpenCV,分别实现 Python 与 C++ 环境的快速上手。 适用平台:Windows 10/11(本文以 Windows 为主要示范,Linux 或 macOS 用户可参照各自系统的包管理细节进行适当调整)。 摘要 本文为新手用户提供了最全的 VSCode & PyCharm 配置 OpenCV 教程,涵盖 Python 与

By Ne0inhk
【C++篇】面向对象编程的三大特性:深入解析继承机制

【C++篇】面向对象编程的三大特性:深入解析继承机制

目录 一、继承的概念  二、继承的基本定义 2.1 继承的定义格式 2.2 三大继承方式与访问限定符 三、基类与派生类的对象赋值转换 3.1 合法的赋值转换 小tip:子类对象赋值给父类对象不会产生临时变量 3.2 非法的赋值转换 3.3 强制类型转换的注意事项(了解) 四、继承中的作用域 4.1 成员变量的隐藏 4.2 成员函数的隐藏 五、派生类的默认成员函数 5.1 核心规则 5.2 代码演示 问题:为何析构函数的调用顺序是:派生类、基类? 六、继承的特殊场景:友元与静态成员 6.1

By Ne0inhk
【C++】一篇文章了解C++的异常处理机制

【C++】一篇文章了解C++的异常处理机制

异常 基本异常处理关键字 在 C++ 中,异常处理是一种机制,用于处理程序在运行时发生的异常情况。异常是指程序执行期间发生 的意外事件,比如除以零、访问无效的内存地址等。通过使用异常处理机制,可以使程序更健壮,并能 够处理这些意外情况,避免程序崩溃或产生不可预测的结果。 在 C++ 中,异常处理通常包括以下关键词和概念: * try-catch 块: try 块用于标识可能会引发异常的代码块,而 catch 块用于捕获和处理异常。 catch 块可以针对不同类型的异常进行处理。 * throw 关键词: throw 用于在程序中显式抛出异常。当发生异常情况时,可以使用 throw 来抛出 一个特定的异常类型。 * 异常类型:异常可以是任何类型的数据,但通常是标准库中的异常类或自定义的异常类。标准库提 供了一些常见的异常类,如 std::exception 及其派生类,用于表示不同类型的异常情况。 核心语法: 关键字作用关键注意点throw中断当前代码流程,

By Ne0inhk