数据结构之链表OJ练习检验你的链表水平是否合格
Posted 一个山里的少年
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了数据结构之链表OJ练习检验你的链表水平是否合格相关的知识,希望对你有一定的参考价值。
目录
1.反转链表
对应letecode链接:
给定单链表的头节点 head ,请反转链表,并返回反转后的链表的头节点。
示例 1:
输入:head = [1,2,3,4,5]
输出:[5,4,3,2,1]
示例 2:
输入:head = [1,2]
输出:[2,1]
示例 3:输入:head = []
输出:[]提示:
链表中节点的数目范围是 [0, 5000]
-5000 <= Node.val <= 5000进阶:链表可以选用迭代或递归方式完成反转。你能否用两种方法解决这道题?
注意:本题与主站 206 题相同: https://leetcode-cn.com/problems/reverse-linked-list/
解题分析: 方法1,双指针法:
设置两个链表:ans 表示已经被反转的链表, head 表示还未被反转的链表。
遍历链表,每次将未被反转的链表 head 的 首元节点反转,直至未被反转的链表为空。
对应代码:
class Solution public: ListNode* reverseList(ListNode* head) ListNode*ans=nullptr; while(head) ListNode*restList =head->next;//保存head的下一个节点 head->next=ans;//翻转指针 ans=head;//迭代 head=restList; return ans; ;
运行结果:
方法二:递归
思想与上面的思想大致相同,翻指针:
1.使用递归函数,一直递归到链表的最后一个结点,该结点就是反转后的头结点,记作 ret .
2.此后,每次函数在返回的过程中,让当前结点的下一个结点的 next->next 指针指向当前节点。
3.同时让当前结点的 next->next 指针指向 NULL ,从而实现从链表尾部开始的局部反转
4.当递归函数全部出栈后,链表反转完成。
图解:
对应代码:
class Solution public: ListNode* reverseList(ListNode* head) if(!head||!head->next)return head; ListNode*ret=reverseList(head->next); head->next->next=head; head->next=nullptr; return ret; ;
方法三:头插法
1.原链表的头结点就是反转之后链表的尾结点,使用 head 标记 .
2.定义指针 cur,初始化为 head .
3.每次都让 head 下一个结点的 next 指向 cur ,实现一次局部反转
4.局部反转完成之后,cur和 head 的 next 指针同时 往前移动一个位置
5.循环上述过程,直至 cur 到达链表的最后一个结点 。
对应代码:
class Solution public: ListNode* reverseList(ListNode* head) if (head == NULL) return NULL; ListNode* cur = head; while (head->next != NULL) ListNode* t = head->next->next; head->next->next = cur; cur = head->next; head->next = t; return cur; ;
递归写法太复杂在这里就不写了:
K个一组翻转链表
对应letecode链接:
给你一个链表,每 k 个节点一组进行翻转,请你返回翻转后的链表。
k 是一个正整数,它的值小于或等于链表的长度。
如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。
进阶:
你可以设计一个只使用常数额外空间的算法来解决此问题吗?
你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。
示例 1:
输入:head = [1,2,3,4,5], k = 2
输出:[2,1,4,3,5]
示例 2:
输入:head = [1,2,3,4,5], k = 3
输出:[3,2,1,4,5]
示例 3:输入:head = [1,2,3,4,5], k = 1
输出:[1,2,3,4,5]
示例 4:输入:head = [1], k = 1
输出:[1]
提示:列表中节点的数量在范围 sz 内
1 <= sz <= 5000
0 <= Node.val <= 1000
1 <= k <= sz
相信有了上面那两道题的基础这道题就每这么难了:
解题思路:
我们需要把链表节点按照 k 个一组分组,所以可以使用一个指针 head 依次指向每组的头节点。这个指针每次向前移动 k 步,直至链表结尾。对于每个分组,我们先判断它的长度是否大于等于 k。若是,我们就翻转这部分链表,否则不需要翻转。
接下来的问题就是如何翻转一个分组内的子链表。翻转一个链表并不难,过程可以参考上面的翻转链表。但是对于一个子链表,除了翻转其本身之外,还需要将子链表的头部与上一个子链表链接,以及子链表的尾部与下一个子链表链接。如下图所示:
对应代码:
翻转[head,tail)区间的链表:
ListNode*Reverse(ListNode*head,ListNode*tail)//翻转区间[head,tail)不包括tail ListNode*cur=head; ListNode*prev=nullptr; while(cur!=tail) ListNode*next=cur->next; cur->next=prev; prev=cur; cur=next; return prev;
总代码:
class Solution public: ListNode*Reverse(ListNode*head,ListNode*tail)//翻转区间[head,tail)不包括tail ListNode*cur=head; ListNode*prev=nullptr; while(cur!=tail) ListNode*next=cur->next; cur->next=prev; prev=cur; cur=next; return prev; ListNode* reverseKGroup(ListNode* head, int k) if(!head||!head->next)return head;//空或者只有一个元素了则返回 ListNode*tail=head; for(int i=0;i<k;i++) if(!tail)//不足k个 return head; tail=tail->next; ListNode*newnode=Reverse(head,tail);//翻转每一组链表 head->next=reverseKGroup(tail,k);//翻转之后head已经变成了尾节点了 return newnode;//翻转完之后返回 ;
这道题和两两一组翻转链表是一样的:
给定一个链表,两两交换其中相邻的节点,并返回交换后的链表。
你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。
示例 1:
输入:head = [1,2,3,4]
输出:[2,1,4,3]
示例 2:输入:head = []
输出:[]
示例 3:输入:head = [1]
输出:[1]
提示:
链表中节点的数目在范围 [0, 100] 内
0 <= Node.val <= 100
进阶:你能在不修改链表节点值的情况下解决这个问题吗?(也就是说,仅修改节点本身。)
由于上图已经讲过思路在这里就不赘述了:
class Solution public: ListNode*Reverse(ListNode*head,ListNode*tail) ListNode*cur=head; ListNode*prev=nullptr; while(cur!=tail) ListNode*next=cur->next; cur->next=prev; prev=cur; cur=next; return prev; ListNode* swapPairs(ListNode* head) if(!head||!head->next)return head; ListNode*tail=head; for(int i=0;i<2;i++) if(!tail)return head; tail=tail->next; ListNode*newnode=Reverse(head,tail); head->next=swapPairs(tail); return newnode; ;
当然还可以采用另外一种递归方式:
1.递归的终止条件是链表中没有节点,或者链表中只有一个节点,此时无法进行交换。
2.如果链表中至少有两个节点,则在两两交换链表中的节点之后,原始链表的头节点变成新的链表的第二个节点,原始链表的第二个节点变成新的链表的头节点。链表中的其余节点的两两交换可以递归地实现。在对链表中的其余节点递归地两两交换之后,更新节点之间的指针关系,即可完成整个链表的两两交换。
3.用 head 表示原始链表的头节点,新的链表的第二个节点,用 newHead 表示新的链表的头节点,原始链表的第二个节点,则原始链表中的其余节点的头节点是 newHead.next。令 head.next = swapPairs(newHead.next),表示将其余节点进行两两交换,交换后的新的头节点为 head 的下一个节点。然后令 newHead.next = head,即完成了所有节点的交换。最后返回新的链表的头节点 newHead。
对应代码:
class Solution public: ListNode* swapPairs(ListNode* head) if (head == nullptr || head->next == nullptr) return head; ListNode* newHead = head->next; head->next = swapPairs(newHead->next); newHead->next = head; return newHead; ;
链表的中间节点:
对应letecode链接:
题目描述:
给定一个头结点为 head 的非空单链表,返回链表的中间结点。
如果有两个中间结点,则返回第二个中间结点。
示例 1:
输入:[1,2,3,4,5]
输出:此列表中的结点 3 (序列化形式:[3,4,5])
返回的结点值为 3 。 (测评系统对该结点序列化表述是 [3,4,5])。
注意,我们返回了一个 ListNode 类型的对象 ans,这样:
ans.val = 3, ans.next.val = 4, ans.next.next.val = 5, 以及 ans.next.next.next = NULL.
示例 2:输入:[1,2,3,4,5,6]
输出:此列表中的结点 4 (序列化形式:[4,5,6])
由于该列表有两个中间结点,值分别为 3 和 4,我们返回第二个结点。提示:
给定链表的结点数介于 1 和 100 之间。
解题思路:快慢指针
1.用两个指针
slow
与fast
一起遍历链表。
2.slow
一次走一步,fast
一次走两步。那么当fast
到达链表的末尾时,slow
必然位于中间位置。3.注意链表的长度有可能是偶数或者奇数,所以循环结束的条件为fast!=nullptr&&fast->next!=nullptr;
对应代码:
class Solution public: ListNode* middleNode(ListNode* head) ListNode*fast=head; ListNode*slow=head; while(fast&&fast->next) fast=fast->next->next; slow=slow->next; return slow; ;
提交结果:
l
链表倒数第k个节点:
对应letecode 链接:
题目描述:
输入一个链表,输出该链表中倒数第k个节点。为了符合大多数人的习惯,本题从1开始计数,即链表的尾节点是倒数第1个节点。
例如,一个链表有 6 个节点,从头节点开始,它们的值依次是 1、2、3、4、5、6。这个链表的倒数第 3 个节点是值为 4 的节点。
示例:
给定一个链表: 1->2->3->4->5, 和 k = 2.
返回链表 4->5.
解题思路:快慢指针
1.定义一个快指针先让它走k步
2.定义一个慢指针指向链表的头节点,此时和fast同时走。
3.结束条件,当fast走到空了就结束了,此时slow只是倒数第k个节点
对应代码:
class Solution public: ListNode* getKthFromEnd(ListNode* head, int k) ListNode*fast=head; for(int i=0;i<k;i++) if(fast)fast=fast=fast->next;//防止链表的长度不够走 ListNode*slow=head; while(fast) fast=fast->next;//同时走 slow=slow->next; return slow; ;
删除链表倒数第k个节点:
对应letecode链接:
剑指 Offer II 021. 删除链表的倒数第 n 个结点 - 力扣(LeetCode) (leetcode-cn.com)
解题思路:快慢指针
1.这题与上题思路大致相同,只是本题是要删除倒数第k个节点,而上题是只要我们找到倒数第k个节点。
2.同样的我们可以 定义fast,slow指针先让fast指针先走K+1部,在让slow和fast同时走但是,循环结束之后我们要删除的刚好是slow的位置,那么我们就必须知道slow的前一个让后在将slow指向的节点删除,返回头节点即可
但是如果我们考虑特殊情况:
考虑一种比较特殊的情况,若需要删除的就是头节点,这时候就存在 bug 。如上图中快指针先走了 6 步,这时候已经超出了尾节点,所以对于这种情况需要做 if 特殊判断。一种比较好的避免这种特殊判断的方法是引入哨兵节点 dummy,该结点的 next 指针指向头指针,之后这样处理以 dummy 为头结点的链表就可以规避上述问题,最后返回dummy 的 next 指针所指的链点就是需要的结果。以 3 个结点的链表处理删除头结点的情况为例子,如下图(红色节点为哨兵节点)
对应代码:
class Solution public: ListNode* removeNthFromEnd(ListNode* head, int n) ListNode*dummyHead=new ListNode(-1); dummyHead->next=head; ListNode*fast=head; for(int i=0;i<k;i++)//让快指针走k步 if(fast)fast=fast->next; ListNode*slow=dummyHead; while(fast) fast=fast->next; slow=slow->next; ListNode*del=slow->next; slow->next=slow->next->next; delete del;//删除节点 ListNode*ans=dummyHead->next;//保存头节点 delete dummyHead;//释放哑节点 return ans; ;
运行结果:
合并两个有序链表:
对应letecode链接:
题目描述:
将两个升序链表合并为一个新的 升序 链表并返回。新链表是通过拼接给定的两个链表的所有节点组成的。
示例 1:
输入:l1 = [1,2,4], l2 = [1,3,4]
输出:[1,1,2,3,4,4]
示例 2:输入:l1 = [], l2 = []
输出:[]
示例 3:输入:l1 = [], l2 = [0]
输出:[0]提示:
两个链表的节点数目范围是 [0, 50]
-100 <= Node.val <= 100
l1 和 l2 均按 非递减顺序 排列
解题思路:
1.我们可以定义一个哑节点prevHead。
2.假设两个链表分别为l1,和l2,变量l1和l2,去它们之中小的那个尾插到哑节点(prevHead)后面,在将对应链表中的节点向后移一位 .
3.当循环结束之后l1和l2中有一个链表的节点取完了但是我们不知道是那一个节点走完了,但是我们可以都判断一个,将其链接到新链表的尾步即可。
4.同时我们还需要定义一个prev来记录新链表的头,代替哑节点走上述的过程
对应图解:
对应代码:
class Solution public: ListNode* mergeTwoLists(ListNode* l1, ListNode* l2) ListNode*prevHead=new ListNode(-1); ListNode*prev=prevHead; while(l1&&l2) if(l1->val<=l2->val) prev->next=l1; prev=prev->next; l1=l1->next; else prev->next=l2; prev=prev->next; l2=l2->next; if(l1)prev->next=l1; if(l2)prev->next=l2; return prevHead->next; ;
运行结果:
链表插入排序:
对应letecode链接:
题目描述:
对链表进行插入排序。
插入排序的动画演示如上。从第一个元素开始,该链表可以被认为已经部分排序(用黑色表示)。
每次迭代时,从输入数据中移除一个元素(用红色表示),并原地将其插入到已排好序的链表中。插入排序算法:
插入排序是迭代的,每次只移动一个元素,直到所有元素可以形成一个有序的输出列表。
每次迭代中,插入排序只从输入数据中移除一个待排序的元素,找到它在序列中适当的位置,并将其插入。
重复直到所有输入数据插入完为止。示例 1:
输入: 4->2->1->3
输出: 1->2->3->4
示例 2:输入: -1->5->3->4->0
输出: -1->0->3->4->5
解题思路:
如果对插入排序不太理解的铁子,可以看一下我写的排序的博客
对应链接:
1.链表与数组的插入排序不同,数组支持随机访问。而链表是不支持随机访问的。我们在数组中可以随意的前后移动,将指针指向值和新元素的值比较后,将新元素插入到合适的位置。我们知道链表查询元素的时间复杂度为 O(n),我们只能够通过遍历链表查询元素。那么我们怎么才能将新元素放到合适的位置呢?
此时我们不能通过移动绿色指针来寻找 5 的合适位置,那么我们应该怎么做了?
当我们发现绿色指针的值大于新元素时(7 > 5),我们则可以定义一个新指针,让其从哑节点开始遍历,直到找到新元素(5)的位置,(4 和 7 之间),然后再将新元素插入即可
通过上面的分析我们知道了大致过程,那么我们的是如何将新元素插入到指定位置的呢?
我们想要将 3 插入到 2 和 4 的中间,此时我们三个指针分别指向 2,4,3
我们共分 4 步,来完成这个操作,见下图
完成上述操作之后链表就变成了:
对应代码:
class Solution public: ListNode* insertionSortList(ListNode* head) if(!head||!head->next)return head;//如果只有一个元素或者链表为空则返回head ListNode*dummyNode=new ListNode(-1);//定义哑节点 dummyNode->next=head; ListNode*prev=head->next; ListNode*last=head; ListNode*temphead=dummyNode; while(prev) if(last->val<=prev->val) last=last->next; prev=prev->next; continue; temphead=dummyNode; while(temphead->next->val<=prev->val)//比较找到对应的位置 temphead=temphead->next; //找到了对应的插入位置,如果不清除的老铁可以自己画一下图 last->next=prev->next; prev->next=temphead->next;//链接 temphead->next=prev; prev=last->next;//迭代往后走 return dummyNode->next; ;
当然我们也可以不用头节点,但是那样就可以复杂一些:在这里就简单的改成代码不过多的解释:
链表排序:
对应letecode链接:
题目描述:
给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表 。
进阶:
你可以在 O(n log n) 时间复杂度和常数级空间复杂度下,对链表进行排序吗?
示例 1:
输入:head = [4,2,1,3]
输出:[1,2,3,4]
示例 2:
输入:head = [-1,5,3,4,0]
输出:[-1,0,3,4,5]
示例 3:输入:head = []
输出:[]
解题思路:归并排序
归并排序,逃脱不了,分合。 思路如下:
分割 -> 通过递归不断分割链表,在此过程中需要保证链表不丢失的情况下,不断想下切割 (e.g. 8 -> 4 -> 2)
关键在于找到链表的中心点, 并从中心点将链表分割成 2 部分。
我们可以使用经典的快慢双指针链表分割方法,其中有个点需要注意,链表长度为奇偶,切割处理方式是不同的,这里根据大家喜欢的方式处理即可,这里没有明确规定必须使用什么切割方式,笔者的切换策略如下:
快指针每次移动 2 步,慢指针每次移动 1 步,当快指针到达链表末尾时,慢指针指向的链表节点即为链表的中点。
找到中点后,将链表进行断开,将当前链表分成 2 部分
对两个链表分别排序
慢指针的下一节点,指向空即可
分割阶段结束 -> 递归退出 -> 直到分割的链表长度为 1
此时递归到底了
merge 环节,退出递归的过程中,不断的排序合并
merge 节点其实包含在分割阶段里边
merge -> 排序当前链表
对应代码:
分割代码:
ListNode*splitListNode(ListNode*head) if(!head||!head->next)return head; ListNode*slow=head; ListNode*fast=head; ListNode*prev=nullptr; while(fast&&fast->next) prev=slow; fast=fast->next->next; slow=slow->next; return prev;
注意中间那个节点要分给对二段链表:
如果只有两个节点的话,slow指向的就是第二个节点。如果将其分给第一个合并的链表那么就会死循环!!!!
合并链表:
istNode*mergeListNode(ListNode*head1,ListNode*head2) ListNode*dummyHead=new ListNode(-1); ListNode*ans=dummyHead; while(head1&&head2) if(head1->val<=head2->val) ans->next=head1; ans=ans->next; head1=head1->next; else ans->next=head2; ans=ans->next; head2=head2->next; if(head1)ans->next=head1; if(head2)ans->next=head2; return dummyHead->next;
由于上面的之前都已经讲解过了在这里就不重复了:
总代码:
class Solution public: ListNode*splitListNode(ListNode*head) if(!head||!head->next)return head; ListNode*slow=head; ListNode*fast=head; ListNode*prev=nullptr; while(fast&&fast->next) prev=slow; fast=fast->next->next; slow=slow->next; return prev; ListNode*mergeListNode(ListNode*head1,ListNode*head2) ListNode*dummyHead=new ListNode(-1); ListNode*ans=dummyHead; while(head1&&head2) if(head1->val<=head2->val) ans->next=head1; ans=ans->next; head1=head1->next; else ans->next=head2; ans=ans->next; head2=head2->next; if(head1)ans->next=head1; if(head2)ans->next=head2; return dummyHead->next; ListNode* sortList(ListNode* head) if(!head||!head->next)return head; ListNode*mid=splitListNode(head); ListNode*midNext=mid->next; mid->next=nullptr; ListNode*head1=sortList(head); ListNode*head2= sortList(midNext); return mergeListNode(head1,head2); ;
运行结果:
当然这题还可以使用快速排序的前后指针法进行排序:
我们可以选取头节点作为基准值,遍历链表,将比它小的节点头插在它前面,比它大的节点尾插在它后面
假设lhead维护的是小于基准值的头插指针,utail维护的是大于等于基准值的尾插指针
则一次对[head , end)快排结束后有
-[ lhead , head ) (左闭右开)是小于基准值的一部分
-[ head.next , end ) (左闭右开)是大于等于基准值的一部分
再分治这两部分即可
对应代码:s
class Solution public: ListNode* sortList(ListNode* head) return quickSortListNode(head,nullptr); ListNode*quickSortListNode(ListNode*head,ListNode*end) if(head ==end || head->next ==end) return head; ListNode* lhead = head ,*utail = head ,*p = head->next; while (p != end) ListNode *next = p->next; if(p->val < head->val)//头插 p->next = lhead; lhead = p; else //尾插 utail->next = p; utail = p; p = next; utail->next = end; ListNode *node = quickSortListNode(lhead, head); head->next = quickSortListNode(head->next, end); return node; ;
删除链表的节点:
对应letecode链接:
题目描述:
给定单向链表的头指针和一个要删除的节点的值,定义一个函数删除该节点。
返回删除后的链表的头节点。
注意:此题对比原题有改动
示例 1:
输入: head = [4,5,1,9], val = 5
输出: [4,1,9]
解释: 给定你链表中值为 5 的第二个节点,那么在调用了你的函数之后,该链表应变为 4 -> 1 -> 9.
示例 2:输入: head = [4,5,1,9], val = 1
输出: [4,5,9]
解释: 给定你链表中值为 1 的第三个节点,那么在调用了你的函数之后,该链表应变为 4 -> 5 -> 9.说明:
题目保证链表中节点的值互不相同
若使用 C 或 C++ 语言,你不需要 free 或 delete 被删除的节点
解题思路:
题中说了链表中的节点值互不相同,也就是说最多只能删除一个。最简单的一种方式就是双指针求解,我们使用两个指针一个指向当前的节点,一个指向当前的上一个节点。
对应代码:
class Solution public: ListNode* deleteNode(ListNode* head, int val) ListNode*dummyListNode=new ListNode(-1); ListNode*prev=dummyListNode; dummyListNode->next=head; ListNode*cur=head; while(cur) if(cur->val==val) prev->next=cur->next;//删除节点 cur=cur->next; break; else cur=cur->next; prev=prev->next;//同时往后走 ListNode*ans=dummyListNode->next;//保存答案 delete dummyListNode; return ans; ;
对应运行结果:
删除链表的节点II:
对应letecode链接:
题目描述:
存在一个按升序排列的链表,给你这个链表的头节点 head ,请你删除链表中所有存在数字重复情况的节点,只保留原始链表中 没有重复出现 的数字。
返回同样按升序排列的结果链表。
示例 1:
输入:head = [1,2,3,3,4,4,5]
输出:[1,2,5]
示例 2:
输入:head = [1,1,1,2,3]
输出:[2,3]提示:
链表中节点数目在范围 [0, 300] 内
-100 <= Node.val <= 100
题目数据保证链表已经按升序排序
解题思路:
题目与上题的不同之处是,删除所有重复出现的元素。如示例所示,头结点是1,其后结点和其重复,因此也要删除。这时,用解决上题题的思路就不合适了。
因此,需要一个虚拟头结点,然后用变量prev指向该虚拟头结点。这样在删除重复结点之后,剩余的结点就可以挂在prev之后继续考察了。具体步骤我们一起看下。
为了方便演示,我将示例给出的链表删减如下:
然后变量difNode指向cur所指向的结点,用以记录和当前考察结点不同的结点位置。
变量curRepeatNum表示和变量cur指向的结点重复的结点个数,初始值为0
这时,变量cur和变量difNode指向的是同一个结点,因此curRepeatNum=1。
接着,将变量difNode向后移动一个位置,看下一个结点和变量cur指向的结点值是否相等。在这里,变量cur和变量difNode指向的结点值相等,因此curRepeatNum=2。
接着,将变量difNode继续向后移动一个位置,看下一个结点和变量cur指向的结点值是否相等。在这里,变量cur和变量difNode指向的结点值不相等。
时curRepeatNum=2,表示cur指向的结点1在链表中出现了2次。接着要做的是将变量prev指向的结点的后继指针指向变量difNode所指向的结点。这样,将就重复结点1从链表中删除了。
最后,要做的是将变量cur指向difNode所指向的结点,进行下一个结点的去重
对应代码:
class Solution public: ListNode* deleteDuplicates(ListNode* head) ListNode*dummyHead=new ListNode(-1); dummyHead->next=head; ListNode*prev=dummyHead; ListNode*cur=head; while(cur) int curRepeatNum=0; ListNode*difNode=cur;// 找到和cur指向的结点值不同的结点 while(difNode&&difNode->val==cur->val) curRepeatNum++; difNode=difNode->next; if(curRepeatNum>1)// 如果curRepeatNum的值大于1,则表示cur指向的结点重复出现了 prev->next=difNode; else prev=cur;// cur指向的结点没有重复出现,则将变量prev指向cur所指向的结点 cur=difNode; return dummyHead->next; ;
运行结果:
环形链表(重点面试常考)
对应letecode链接:
141. 环形链表 - 力扣(LeetCode) (leetcode-cn.com)https://leetcode-cn.com/problems/linked-list-cycle/
题目描述:
给你一个链表的头节点 head ,判断链表中是否有环。
如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。注意: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 或者链表中的一个 有效索引 。
解题思路:快慢指针:
最简单的一种方式就是快慢指针,慢指针每次走一步,快指针每次走两步,如果相遇就说明有环,如果有一个为空说明没有环。代码比较简单:
class Solution public: bool hasCycle(ListNode *head) ListNode*fast=head; ListNode*slow=head; while(fast&&fast->next) fast=fast->next->next; slow=slow->next; if(slow==fast)return true; return false; ;
但是如果这道题是在面试中就不会这么容易让你过,因为太简单:面试官一般都会问你为什么他们一定会相遇,fast每次只能走两步吗?一次走三步走四步行不行,或者走n步行不行
答案是fast每次走两步才能保证他们一定会相遇 待博主细细道来:
假如环的长度是m,快慢指针最近的间距是n,如图所示
快指针每次走两步,慢指针每次走一步,所以每走一次快慢指针的间距就要缩小一步,他们之间的间距没走一次缩小1步由于m,n都是整数那么迟早都会减到0.
在图一中当走n次的时候就会相遇,在图二中当走m-n次的时候就会相遇。
那如果fast每次走三步了?还会不会相遇了?答案是不一定?
还是一样的当slow进环之后,假设他们的间距为n,但是fast,slow每走一次距离会缩小2,此时如果n是偶数那么它们会相遇,如果是奇数,那么他们之间的距离会减到-1,-1代码什么意思了?-1代码fast反超slow此时他们之间的距离就变成了环的长度减1,也就是m-1,和上面的分析一样,如果m-1是偶数那么就可以相遇当时如果m-1是奇数那么它又会减到-1,那么就会重复上述步骤。那么他们也就永远不会相遇。
fast一次走四步的分析方法也是类似的,各位铁子可以自己下去推导
环形链表Il(重点面试常考)
对应letecode链接:
对应题目描述:
给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。
如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。注意:pos 不作为参数进行传递,仅仅是为了标识链表的实际情况。
不允许修改 链表。
示例 1:
输入:head = [3,2,0,-4], pos = 1
输出:返回索引为 1 的链表节点
解释:链表中有一个环,其尾部连接到第二个节点。
示例 2:输入:head = [1,2], pos = 0
输出:返回索引为 0 的链表节点
解释:链表中有一个环,其尾部连接到第一个节点。
示例 3:输入:head = [1], pos = -1
输出:返回 null
解释:链表中没有环。
提示:
链表中节点的数目范围在范围 [0, 104] 内
-105 <= Node.val <= 105
pos 的值为 -1 或者链表中的一个有效索引
解题思路:双指针
1.设slow fast 第一次相遇 。设两指针
fast
,slow
指向链表头部head
,fast
每轮走 2 步,slow
每轮走 11步。第一种情况:
fast 指针走过链表末端,说明链表无环,直接返回 null;
第二种情况:若有环,两指针一定会相遇。因为每走 1 轮,fast 与 slow 的间距 -1,fast 终会追上 slow
第三种情况:
当fast == slow时, 两指针在环中 第一次相遇 。下面分析此时fast 与 slow走过的 步数关系 :
设链表共有 a+b 个节点,其中 链表头部到链表入口 有 a 个节点(不计链表入口节点), 链表环 有 b 个节点(注意:a 和 b 是未知数,例如图解上链表 a=4 , b=5)。
设两指针分别走了 f,s 步,则有:
fast 走的步数是slow步数的 2 倍,即 f = 2s;(解析: fast 每轮走 2 步)
fast 比 slow多走了 n 个环的长度,即f=s+nb;( 解析: 双指针都走过 a 步,然后在环内绕圈直到重合,重合时 fast 比 slow 多走 环的长度整数倍 );
以上两式相减得:f = 2nb,s = nb,即fast和slow 指针分别走了 2n,n 个 环的周长 (注意: n 是未知数,不同链表的情况不同)。
如果让指针从链表头部一直向前走并统计步数k,那么所有 走到链表入口节点时的步数 是:k=a+nb(先走 a 步到入口节点,之后每绕 1 圈环( b 步)都会再次到入口节点)。
而目前,slow 指针走过的步数为 nbnb 步。因此,我们只要想办法让 slow 再走 a 步停下来,就可以到环的入口。
但是我们不知道 a 的值,该怎么办?依然是使用双指针法。我们构建一个指针,此指针需要有以下性质:此指针和slow 一起向前走 a 步后,两者在入口节点重合。那么从哪里走到入口节点需要 a 步?答案是链表头部head。双指针第二次相遇:
slow指针 位置不变 ,将fast指针重新 指向
以上是关于数据结构之链表OJ练习检验你的链表水平是否合格的主要内容,如果未能解决你的问题,请参考以下文章