今天是12月的第一天,刷了两个题
JZ22 链表中倒数最后k个结点
解法1 队列法(初级)
初级是因为时间复杂度O(n) 空间复杂度O(n)
思想简单:直接用数组把每一个节点存起来,输出第n-k个
class Solution {
public:
ListNode* FindKthToTail(ListNode* pHead, int k) {
vector<ListNode*> res;
while(pHead){
res.push_back(pHead); //把每一个节点存起来
pHead = pHead->next; //遍历完整个链表
}
if(res.size()>=k){
return res[res.size()-k];//如果总节点个数大于等于k,就输出倒数第k个
}
return nullptr;
}
};
解法2 双指针法(进阶)
这个就好一些了:时间复杂度O(n)空间复杂度O(1)
思想稍微难一些:
- 用两个指针指向头节点,先让走得快的指针移动k步
- 两个指针一次走一步,直到走得快的先走完整个链表
- 走的满的指针即为所求(倒数第k个)
其中的思想核心就是让两个指针的相对位置永远相差k,当后面的指针指向队尾的后面,前面的指针就是指向倒数第k个。
/**
* struct ListNode {
* int val;
* struct ListNode *next;
* ListNode(int x) : val(x), next(nullptr) {}
* };
*/
class Solution {
public:
ListNode* FindKthToTail(ListNode* pHead, int k) {
ListNode* fast=pHead;ListNode* slow=pHead;
for(int i=0;i<k;i++){ //让fast节点先走k步
if(!fast) return nullptr;
fast = fast->next;
}
while(fast){ //保持k的相对距离直到结尾
fast=fast->next;
slow = slow->next;
}
return slow;
}
};
JZ18 删除链表的节点
这个题挺简单,数据结构老师也详细讲过。
链表删除节点的本质就是,让当前要删除的结点的前面那个结点的next,指向当前结点的next。当前节点就被绕过去了,也就是从链表上被删除了。
class Solution {
public:
ListNode* deleteNode(ListNode* head, int val) {
ListNode* tmp = head;
ListNode* last = head;
if(head->val==val) return head->next;
tmp = tmp->next;
while(tmp){
if(tmp->val==val){
last->next = tmp->next; //核心语句,解释在上面
return head;
}
tmp=tmp->next;
last=last->next;
}
return head;
}
};