链表-快慢指针(C++)

article/2025/11/10 0:41:56

一、链表

链表是由一组在内存中不必相连(不必相连:可以连续也可以不连续)的内存结构Node,按特定的顺序链接在一起的抽象数据类型。

我们常见的链表结构有单链表和双向链表。

单链表,保存了下一个结点的指针,可以根据下一结点指针向下遍历

双向链表,保存了向上一个结点和向下一个结点的指针,所以可以向上下两个方向遍历。

单链表结构一般如下:

template<typename T>
struct Node
{T data; //数据Node* next; //指向下一个结点的指针
};

双向链表的结构一般如下:

template<typename T>
struct Node
{T data; //数据Node* previous; //指向上一个结点的指针Node* next; //指向下一个结点的指针
};

在算法中,我们遇到的链表题目一般可以使用一些经典的思路求解。本篇博客主要讨论这些经典的链表解题思路之一,快慢指针。

二、快慢指针

快慢指针也称龟兔赛跑算法,又叫判圈算法。具体方法是声明两个指针fast 指针和slow 指针,这两个指针按照题目的需求有不同的行进方案(例如fast每次行进2步,slow每次行进1步),但基本上步长是不同的,所以叫龟兔赛跑算法。

比较经典的题目如下,判断链表是否有环,所以又叫判圈算法。

【题目】环形链表

判断一个单链表是否有环,并且返回第一个入环节点。

一个单链表有环,即如下形式,图中第一个入环节点为2

<算法简述>

使用快慢指针求解这个问题。我们使fast和slow同时从头结点出发,同频行进。

其中fast每次走2个结点,slow每次走一个结点。

如果fast或 slow能够为nullptr,则链表无环,如果fast和slow相遇,则链表有环。

这里很容易比较得出快慢指针的一个特点1:可以判圈

使slow指针在原地(相遇位置)保持不动,并使fast指针返回头节点处,二者同频出发,每次均走一步。则再次相遇的节点为第一个入环节点。

<C++代码实现>

ListNode* hasCycle(ListNode* head)
{ListNode* fast = head;ListNode* slow = head;//判断是否有环while (nullptr != fast && nullptr != slow){if (nullptr == fast->next) return nullptr;fast = fast->next->next;slow = slow->next;//无环,返回空指针if (nullptr == fast || nullptr == slow){return nullptr;}if (fast == slow){break;}}//快指针回到头结点,快慢指针每次均走一步,相遇节点为第一个入环节点fast = head;while (nullptr != slow){if (fast == slow){break;}fast = fast->next;slow = slow->next;}return slow;
}

<复杂度分析>

时间复杂度为O(N),额外空间复杂度为O(1)。

【题目】判断一个单链表是否是回文结构

回文结构:正向遍历和反向遍历每个结点都是相同数据的结构。例如1->2->1,1->2->2->1都是回文结构。

【方法一】使用栈结构(对照组)

        想到判断回文结构 ,我们可以想到栈结构,因为栈是符合先入后出的数据结构,所以只要我们先遍历一遍链表,对数据进行压栈。再依次弹出栈顶,就会得到链表的反向输出。这样可以同时取到原链表的正向输出和反向输出,一一比较之后,就可以判断是否回文。

<C++代码实现>

bool isPalindrome(Node* hd)
{stack<int> st;Node* temp = hd;while (nullptr != temp){st.push(temp->data);temp = temp->next;}temp = hd;while (nullptr != temp){if (temp ->data != st.top()){return false;}temp = temp->next;st.pop();}return true;
}

<复杂度分析>

上述额外使用栈结构,所以额外空间复杂度为O(N),时间复杂度为O(N)。

【方法二】快慢指针方法

分析:这个题目一般想不到使用快慢指针方法,但是快慢指针具有一个和此题吻合的特点2:利用快慢指针的不同步数,可以找到链表的中点。如果此题找到中点,我们也可以镜像地比较中点左右的元素,得出是否是回文结构的结论。

而使用下列方法,可以使额外空间复杂度为O(1),这就是这个算法的优势。

bool isPalindrome(Node* hd)
{if (nullptr == hd){return true;}Node* fast = hd->next;Node* slow = hd;//快慢指针同时遍历,快指针每次走2步,慢指针每次走1步//当快指针到终点的时候,慢指针正好处于中点位置while (nullptr != fast && nullptr != fast->next){fast = fast->next->next;slow = slow->next;}//慢指针继续往下遍历,遍历的同时将指向反向Node *temp1 = slow; //temp1用于记录slow的前一个结点,用于逆序链表的后半部分Node *temp2 = slow->next; //temp1用于记录slow的后一个结点,放在链表断开后指针丢失Node *mid = slow;   //保存中点结点,便于后续使用while (nullptr != slow){slow->next = temp1;temp1 = slow;slow = temp2;if (nullptr != slow){temp2 = slow->next;}}//此时temp1为原链表终点结点,使temp2指向头结点,同时遍历temp1,temp2比较数值Node *ed = temp1; //保存终点结点,便于后续还原链表temp2 = hd;bool ret = true; //是否是回文结构while (temp1 != mid){if (temp1->data != temp2->data){ret = false;break;}temp1 = temp1->next;temp2 = temp2->next;}//从终点开始,还原原链表temp1 = ed->next;ed->next = nullptr;while (mid != ed){temp2 = ed;ed = temp1;temp1 = ed->next;ed->next = temp2;}return ret;
}

<复杂度分析>

可见此算法的coding复杂很多,但是额外空间复杂度可以做到O(1),时间复杂度为O(N)。

【题目】删除链表的倒数第 N 个结点

给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。

<分析>

此题的重点难点是:如何找到倒数第N个结点。

做完上面的判断回文结构的题,我们认识到快慢指针的特点2,应该可以很容易想到:我们使fast先走N个结点,再让slow和fast同频同步行进,当fast为最后一个结点的时候,slow所在结点即为倒数第N个结点。

<C++代码实现>

ListNode* removeNthFromEnd(ListNode* head, int n)
{ListNode* fast = head;ListNode* slow = head;//fast指针先走n步while (nullptr != fast && --n > 0){fast = fast->next;}//如果链表长度小于n,则直接返回头指针if (nullptr == fast){return head;}//如果倒数第n个就是头节点,则换头节点if (nullptr == fast->next){head = head->next;delete slow;return head;}//fast和slow都前进,fast走到倒数第一个,则删除slow->next节点while (nullptr != fast->next->next){slow = slow->next;fast = fast->next;}fast = slow->next;slow->next = slow->next->next;delete fast;return head;
}

三、总结

从上面的题目中,总结下快慢指针的一般特点

1、可以判断链表是否有环,并得出入环节点;

2、可以定位链表中的某个指定位置的结点;

后续在求解问题的过程中,如果还发现其他妙用,博主会持续更新。


http://chatgpt.dhexx.cn/article/bpwUHUj1.shtml

相关文章

面试题 02.08. 环路检测-快慢指针+如何找到环的入口?(证明)Java

1.题目 2.思路 方法一——哈希表记录节点 思路很简单&#xff0c;记录一下每个节点出现的次数&#xff0c;如果某个节点出现了两次&#xff0c;代表此时有环&#xff0c;并且是环的入口&#xff0c;直接返回即可。 时间复杂度O(N) 空间复杂度O(N) public class Solution {…

链表中快慢指针的应用

目录 一、链表的中间结点 二、回文链表 三、链表中倒数第K个结点 四、删除链表的倒数第n个结点 五、环形链表 六、环形链表Ⅱ 一、链表的中间结点 给定一个头结点为 head 的非空单链表&#xff0c;返回链表的中间结点。 如果有两个中间结点&#xff0c;则返回第二个中间…

快慢指针思想

快慢指针思想 在做题当中经常会用到快慢指针&#xff0c;快慢指针就是定义两根指针&#xff0c;移动的速度一快一慢&#xff0c;从而创造出自己想要指针的差值。这个差值可以让我们找到链表上相应的节点。 参考链接&#xff1a;https://www.jianshu.com/p/21b4b8d7d31b 应用 …

指针的运用——快慢指针

快慢指针是指针的一种类型&#xff0c;在这里我们来了解下快慢指针 让我们来看一道题 一、题目 876. 链表的中间结点 首先我们对这道题进行分析&#xff0c;最容易让人想到的方法是直接使用n/2找到中点&#xff0c;如果我们不对链表进行遍历&#xff0c;我们该怎么做呢&…

浅谈快慢指针

快慢指针 快慢指针 快慢指针1.快慢指针的概念&#xff1a;2.快慢指针的应用&#xff1a;1. 判断单链表是否为循环链表2. 在有序链表中寻找中位数3.链表中倒数第k个节点 1.快慢指针的概念&#xff1a; 快慢指针就是存在两个指针&#xff0c;一个快指针&#xff0c;一个慢指针&a…

快慢指针应用总结

快慢指针 快慢指针中的快慢指的是移动的步长&#xff0c;即每次向前移动速度的快慢。例如可以让快指针每次沿链表向前移动2&#xff0c;慢指针每次向前移动1次。 快慢指针的应用 &#xff08;1&#xff09;判断单链表是否存在环 如果链表存在环&#xff0c;就好像操场的跑道是…

十大常用经典排序算法总结!!!

爆肝整理&#xff01;堪称全网最详细的十大常用经典排序算法总结&#xff01;&#xff01;&#xff01; 写在开头&#xff0c;本文经过参考多方资料整理而成&#xff0c;全部参考目录会附在文章末尾。很多略有争议性的细节都是在不断查阅相关资料后总结的&#xff0c;具有一定…

经典五大算法思想-------入门浅析

算法&#xff1a;求解具体问题的步骤描述&#xff0c;代码上表现出来是解决特定问题的一组有限的指令序列。 1、分治&#xff1a; 算法思想&#xff1a;规模为n的原问题的解无法直接求出&#xff0c;进行问题规模缩减&#xff0c;划分子问题&#xff08;这里子问题相互独立而且…

算法设计经典算法

一、贪婪算法 1、概述 贪婪法又称贪心算法&#xff0c;是当追求的目标是一个问题的最优解时&#xff0c;设法把对整个问题的求解工作分成若干步骤来完成&#xff0c;是寻找最优解问题的常用方法。 贪婪法的特点是一般可以快速得到满意的解&#xff0c;因为它省去了为找最优解…

算法之经典图算法

图介绍表示图的数据结构图的两种搜索方式DFS可以处理问题BFS可以处理问题有向图最小生成树最短路径 图介绍 图&#xff1a;是一个顶点集合加上一个连接不同顶点对的边的集合组成。定义规定不允许出现重复边&#xff08;平行边&#xff09;、连接到顶点自身的边&#xff08;自环…

计算机10大经典算法

算法一&#xff1a;快速排序法 快速排序是由东尼霍尔所发展的一种排序算法。在平均状况下&#xff0c;排序 n 个项目要Ο(n log n)次比较。在最坏状况下则需要Ο(n2)次比较&#xff0c;但这种状况并不常见。事实上&#xff0c;快速排序通常明显比其…

算法设计——五大算法总结

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 算法设计总结 一、【分治法】二、【动态规划法】三、【贪心算法】四、【回溯法】五、【分支限界法】 一、【分治法】 在计算机科学中&#xff0c;分治法是一种很重要的算法。…

十大经典算法总结

正文 排序算法说明 &#xff08;1&#xff09;排序的定义&#xff1a;对一序列对象根据某个关键字进行排序&#xff1b; 输入&#xff1a;n个数&#xff1a;a1,a2,a3,...,an 输出&#xff1a;n个数的排列:a1,a2,a3,...,an&#xff0c;使得a1<a2<a3<...<an。 再…

九大经典算法

1. 冒泡排序&#xff08;Bubble Sort&#xff09; 两个数比较大小&#xff0c;通过两两交换&#xff0c;像水中的泡泡一样&#xff0c;较大的数下沉&#xff0c;较小的数冒起来。 算法描述&#xff1a; 1.比较相邻的元素。如果第一个比第二个大&#xff0c;就交换它们两个&a…

最常用的五大算法

一、贪心算法 贪心算法&#xff08;又称贪婪算法&#xff09;是指&#xff0c;在对问题求解时&#xff0c;总是做出在当前看来是最好的选择。也就是说&#xff0c;不从整体最优上加以考虑&#xff0c;他所做出的仅是在某种意义上的局部最优解。贪心算法不是对所有问题都能得到整…

几种经典算法

1.分治法 分治法也叫做分而治之法。核心思想是将一个难以直接解决的大问题依照相同的概念分割成两个或者多个相同的小问题&#xff0c;以便各个击破。 如图所示&#xff1a; 2.递归法 递归法和分而治之法像一对孪生兄弟&#xff0c;都是将一个复杂的算法问题进行分解&#x…

五大常用经典算法—分治算法

原文作者&#xff1a;bigsai 原文地址&#xff1a;五大常用算法&#xff1a;一文搞懂分治算法 目录 前言 分治算法介绍 分治算法经典问题 二分搜索 快速排序 归并排序(逆序数) 最大子序列和 最近点对 结语 前言 分治算法&#xff08;divide and conquer&#xff09;是…

十大经典算法

十大经典排序算法&#xff08;动图演示&#xff09; 0、算法概述 0.1 算法分类 十种常见排序算法可以分为两大类&#xff1a; 非线性时间比较类排序&#xff1a;通过比较来决定元素间的相对次序&#xff0c;由于其时间复杂度不能突破O(nlogn)&#xff0c;因此称为非线性时间比…

OpenX系列标准:OpenDRIVE标准简述

1.概述 ​ 作为一个完整的仿真测试场景描述方案&#xff0c;OpenX系列标准包括&#xff1a;OpenDRIVE、OpenSCENARIO和OpenCRG。 标准文件格式文件内容OpenDRIVE.xodr静态部分&#xff08;如道路拓扑结构、交通标志标线等&#xff09;OpenDRIVE.tdo保存ROD项目时生成的文件&a…

OpenDRIVE坐标系解读

几种坐标系简述 opendrive标准主要包括三种坐标系&#xff1a;inertial(x, y, z)、reference line(s, t, h)、local(u, v, z) 下面这张图片笔者认为还是比较清晰的展示了三种坐标系的关系的&#xff1a; 惯性坐标系&#xff08;Inertial&#xff09; 惯性坐标系最简单&am…