前言
这里记录一下陈菜菜的刷题记录,主要应对25秋招、春招
个人背景
211CS本+CUHK计算机相关硕,一年车企软件开发经验
代码能力:有待提高
常用语言:C++
系列文章目录
第一天 数组 part01
第二天 数组 part02
第三天 链表 part01
第四天 链表 part02
`
文章目录
一、今日任务
● 24. 两两交换链表中的节点
● 19.删除链表的倒数第N个节点
● 面试题 02.07. 链表相交
● 142.环形链表II
● 总结
二、详细布置
24.两两交换链表中的节点
题目链接:力扣24
文章讲解:代码随想录-两两交换链表中的节点
视频讲解:代码随想录
给定一个链表,两两交换其中相邻的节点,并返回交换后的链表。
你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换
提示:
链表中节点的数目在范围 [0, 100] 内
0 <= Node.val <= 100
样例1:
输入:head = [1,2,3,4]
输出:[2,1,4,3]
样例2:
输入:head = []
输出:[]
样例3:
输入:head = [1]
输出:[1]
思路
这题一眼看就是链表的快慢指针!
实战
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode() : val(0), next(nullptr) {}
* ListNode(int x) : val(x), next(nullptr) {}
* ListNode(int x, ListNode *next) : val(x), next(next) {}
* };
*/
class Solution {
public:
ListNode* swapPairs(ListNode* head) {
ListNode dum(0,head);
ListNode* slow=&dum;
ListNode *now=head;
//ListNode* fast=head->next;
while(now && now->next){
ListNode* temp=now->next->next;
slow->next=now->next;
now->next->next=now;
now->next=temp;
slow=now;
now=now->next;
}
return dum.next;
}
};
19.删除链表的倒数第N个节点
题目链接:力扣19题链接
文章讲解:图文讲解
视频讲解:[代码训练营]https://www.bilibili.com/video/BV1vW4y1U7Gf/?vd_source=057d67a9ec119b8e9d783e13f911bd3d
)
给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。
提示:
0 <= index, val <= 1000
请不要使用内置的 LinkedList 库。
调用 get、addAtHead、addAtTail、addAtIndex 和 deleteAtIndex 的次数不超过 2000 。
样例1:
输入:head = [1,2,3,4,5], n = 2
输出:[1,2,3,5]
样例2:
输入:head = [1], n = 1
输出:[]
样例3:
输入:head = [1,2], n = 1
输出:[1]
思路
这题就是快慢指针的变形。让快指针先移动n个,再让快慢指针一起移动,直到快指针移动到链表末端。这时的慢指针就指向了要删除的节点。
实战
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode() : val(0), next(nullptr) {}
* ListNode(int x) : val(x), next(nullptr) {}
* ListNode(int x, ListNode *next) : val(x), next(next) {}
* };
*/
class Solution {
public:
ListNode* removeNthFromEnd(ListNode* head, int n) {
if(head==NULL||head->next==NULL)
return NULL;
ListNode dum=ListNode(0,head);
ListNode* fast=head;
ListNode* now=head;
ListNode* pre=&dum;
for(int i=0;i<n;i++)
fast=fast->next;
while(fast){
pre=pre->next;
now=now->next;
fast=fast->next;
}
ListNode* nxt=now->next;
pre->next=nxt;
delete now;
return dum.next;
}
};
踩坑
ListNode dum=ListNode(0,head);
不是指针,所以不能赋给指针型变量,但是可以将它的地址赋给指针。同时,引用dum的next不能用->,要用.。
面试题 02.07. 链表相交(同160.链表相交)
题目描述
给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表没有交点,返回 null 。
题目数据 保证 整个链式结构中不存在环。
注意,函数返回结果后,链表必须 保持其原始结构
提示:
listA 中节点数目为 m
listB 中节点数目为 n
0 <= m, n <= 3 * 104
1 <= Node.val <= 105
0 <= skipA <= m
0 <= skipB <= n
如果 listA 和 listB 没有交点,intersectVal 为 0
如果 listA 和 listB 有交点,intersectVal == listA[skipA + 1] == listB[skipB + 1]
样例1:
输入:intersectVal = 8, listA = [4,1,8,4,5], listB = [5,0,1,8,4,5], skipA = 2, skipB = 3
输出:Intersected at '8'
解释:相交节点的值为 8 (注意,如果两个链表相交则不能为 0)。
从各自的表头开始算起,链表 A 为 [4,1,8,4,5],链表 B 为 [5,0,1,8,4,5]。
在 A 中,相交节点前有 2 个节点;在 B 中,相交节点前有 3 个节点。
样例2:
输入:intersectVal = 2, listA = [0,9,1,2,4], listB = [3,2,4], skipA = 3, skipB = 1
输出:Intersected at '2'
解释:相交节点的值为 2 (注意,如果两个链表相交则不能为 0)。
从各自的表头开始算起,链表 A 为 [0,9,1,2,4],链表 B 为 [3,2,4]。
在 A 中,相交节点前有 3 个节点;在 B 中,相交节点前有 1 个节点。
样例3:
输入:intersectVal = 0, listA = [2,6,4], listB = [1,5], skipA = 3, skipB = 2
输出:null
解释:从各自的表头开始算起,链表 A 为 [2,6,4],链表 B 为 [1,5]。
由于这两个链表不相交,所以 intersectVal 必须为 0,而 skipA 和 skipB 可以是任意值。
这两个链表不相交,因此返回 null 。
思路
这题思路还是让长一点的链表先移动|m-n|个节点,再让两个链表的指针同时往后移动,直到找到相同的节点。因为如果尾端都相同(相交)的两个链表从交点处开始一定长度相同。
实战
class Solution {
public:
ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {
int cnt=0,m=0,n=0;
int skipA=0,skipB=0;
ListNode* pa=headA;
ListNode* pb=headB;
while(pa){
m++;
pa=pa->next;
}
pa=headA;
while(pb){
n++;
pb=pb->next;
}
pb=headB;
if(m>=n){
for(cnt=0;cnt<m-n;cnt++)
pa=pa->next;
}
else{
for(cnt=0;cnt<n-m;cnt++)
pb=pb->next;
}
while(pa){
if(pa==pb)
return pa;
else{
pa=pa->next;
pb=pb->next;
}
}
return NULL;
}
};
踩坑
1.最后返回的是指针,不是val。不相交的话返回null就行。
142.环形链表II
题目描述
给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。
如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。注意:pos 不作为参数进行传递,仅仅是为了标识链表的实际情况。
不允许修改 链表。
提示:
链表中节点的数目范围在范围 [0, 104] 内
-105 <= Node.val <= 105
pos 的值为 -1 或者链表中的一个有效索引
样例1:
输入:head = [3,2,0,-4], pos = 1
输出:返回索引为 1 的链表节点
解释:链表中有一个环,其尾部连接到第二个节点。
样例2:
输入:head = [1,2], pos = 0
输出:返回索引为 0 的链表节点
解释:链表中有一个环,其尾部连接到第一个节点
样例3:
输入:head = [1], pos = -1
输出:返回 null
解释:链表中没有环
思路
这题是真妙,一开始想用递归去找有没有环发现会死循环,遂放弃。看了讲解才觉得这题真的很巧妙。无论是寻找相遇点,还是从相遇点推环形入口都很巧妙,问题的关键点在从相遇点开始同时移动相同长度的两个指针一定会在环入口处相遇!真的建议大家去看一下讲解,这个推导过程值得一看!
实战
/**
* Definition for singly-linked list.
* struct ListNode {
* int val;
* ListNode *next;
* ListNode(int x) : val(x), next(NULL) {}
* };
*/
class Solution {
public:
ListNode *detectCycle(ListNode *head) {
ListNode* fast=head;
ListNode* slow=head;
while(fast!=NULL&&fast->next!=NULL){
fast=fast->next->next;
slow=slow->next;
if(fast==slow){
ListNode* index1=head;
ListNode* index2=fast;
while(index1!=index2){
index1=index1->next;
index2=index2->next;
}
return index1;
}
}
return NULL;
}
};
踩坑
1.最后返回的是指针,不是val。没有环的话返回null就行。
总结
今天主要学习了链表的一系列操作,
完成刷题4题,142值得多刷。
今天是训练营第四天,今天不上课,就没有那么赶,耐下心把自己不熟练的链表刷完了。其实链表的算法反而没有那么复杂,因为它的遍历必须一个一个查,只是它的数据结构比较繁琐,熟悉了就好理解多了!
加油,坚持打卡的第四天。