【Leetcode】移除链表元素 链表的中间节点 链表中倒数第k个节点

简介: 【Leetcode】移除链表元素 链表的中间节点 链表中倒数第k个节点

一.【Leetcode203】移除链表元素

1.链接

移除链表元素

2.题目再现

A.双指针

1.创建一个指针 cur=head  和一个指针  pre=NULL;  

2.用cur->val 与 val 比较,如果不相等则把 cur 赋给 pre 使cur 指向下一个节点,即

  cur=cur->next

3.如果相等则使 pre 的 next 指向 cur 的 next ,即:

pre->next=cur->next ,然后再 free 掉 cur ,最后再使 cur 等于 pre 的 next,注意在进行这些步骤之前要判断 pre 是否为空 ,若为空即为头删;

演示:

双指针

代码:

1. struct ListNode* removeElements(struct ListNode* head, int val) 
2. {
3. struct ListNode*pre=NULL;
4. struct ListNode*cur=head;
5. while(cur)
6.     {
7. if(cur->val!=val)
8.         {
9.             pre=cur;
10.             cur=cur->next;
11.         }
12. else
13.         {
14. if(pre==NULL)
15.             {
16.                 head=cur->next;
17. free(cur);
18.                 cur=head;
19.             }
20. else
21.             {
22.                 pre->next=cur->next;
23.                 cur=pre->next;
24.             }
25.         }
26.     }
27. return head;
28. }

B.类尾删法

1.创建一个新的指针newhead ,同时为了省去找尾的麻烦,我们可以定义一个尾指针 tail 来保存尾节点;

2.再创建一个指针 cur =head ,用来遍历链表;

3.如果 cur->val != val ,则尾插 ,注意要判断 tail 是否为空 ,类似于单链表的尾插那部分,如果不理解的话,可查看文章 :单链表的增删查改

4.如果 cur->val ==val,则 cur=cur->next ;

5.最后要将尾节点置空。

演示:

类尾插

代码:

1. struct ListNode* removeElements(struct ListNode* head, int val) 
2. {
3. struct ListNode *newhead=NULL;
4. struct ListNode*tail=NULL;
5. struct ListNode*cur=head;
6. while(cur)
7.     {
8. if(cur->val!=val)
9.         {
10. if(tail==NULL)
11.             {
12.                 newhead=tail=cur;
13.             }
14. else
15.             {
16.                 tail->next=cur;
17.                 tail=tail->next;
18.             }
19.             cur=cur->next;
20.         }
21. else
22.         {
23.             cur=cur->next;
24.         }
25.     }
26. if(tail)
27.     {
28.         tail->next=NULL;
29.     }
30. return newhead;
31. }

C.哨兵位

1.malloc 一个哨兵位节点 dummyhead,使其 next 指向 head ;

2.再定义一个节点 tmp = dummyhead ,用这个遍历链表;

3.注意因为 tmp ->next 才是 head ,所以 while 里要写 tmp->next !=NULL

演示:

移除链表元素 哨兵位法动态演示

代码:

1. struct ListNode* removeElements(struct ListNode* head, int val) 
2. {
3. struct ListNode*dummyhead=(struct ListNode*)malloc(sizeof(struct ListNode));
4.     dummyhead->next=head;
5. struct ListNode*tmp=dummyhead;
6. while(tmp->next!=NULL)
7.     {
8. if(tmp->next->val==val)
9.         {
10.             tmp->next=tmp->next->next;
11.         }
12. else
13.         {
14.             tmp=tmp->next;
15.         }
16.     }
17. return dummyhead->next;
18. }

二.【Leetcode876】链表的中间节点

1.链接:链表的中间节点

2.题目再现

3.解法:快慢指针

1.定义一个快指针 fast 和一个慢指针 slow 都初始化为 head;

2.遍历链表,快指针一次走2步,慢指针一次走1步

3.注意:因为链表的长度可能是单数也可能是双数,所以当我们已 fast 是否为NULL 作为循环控制条件的话,要在 fast 走2步前判断 fast->next 是否为空

4.最后慢指针就是中间节点

演示:

链表中间节点 快慢指针动态演示

代码:

1. struct ListNode* middleNode(struct ListNode* head)
2. {
3. struct ListNode*slow=head;
4. struct ListNode*fast=head;
5. while(fast)
6.     {
7. if(fast->next==NULL)  //注意判断
8.         {
9. break;
10.         }
11. else
12.         {
13.             fast=fast->next->next;  //fast 走2步
14.         }
15.         slow=slow->next;   //slow 走1步
16.     }
17. return slow;  //返回慢指针
18. }

三.链表中倒数第k个节点

1.链接:链表中倒数第k个节点

2.题目再现

3.解法 :快慢指针

1.定义一个快指针 fast 和一个慢指针 slow 都初始化为 head;

2.因为倒数第k个节点和尾节点的差为 k-1  ,所以我们先让快指针先走 k-1 步;

或者因为尾节点所指向的NULL 和倒数第k个节点相差k,也可以先让快指针走k步;

这个时候慢指针不动;

3.快指针走完后,快指针和慢指针依次走,每次只走1步

注意,如果是k-1,那么遍历结束的条件是fast->next 是否为空 ,如果是k,那么遍历结束的条件是fast 是否为空;

4.返回慢指针。

演示:

链表倒数第K个节点 快慢指针动态演示

代码:

1. struct ListNode* FindKthToTail(struct ListNode* pListHead, int k )
2. {
3. if(pListHead==NULL)
4.     {
5. return NULL;
6.     }
7. struct ListNode*slow=pListHead;
8. struct ListNode*fast=pListHead;
9. while(k--)  //这里以先走k步为例
10.     {
11. if(fast==NULL)
12.         {
13. return NULL;
14.         }
15.         fast=fast->next;
16.     }
17. while(fast)
18.     {
19.         slow=slow->next;
20.         fast=fast->next;
21.     }
22. return slow;
23. }

😽本篇文章到此就结束了,若有错误或是建议,欢迎小伙伴们指出;😻

😍请多多支持博主哦~🥰

🤩谢谢你的阅读~😃


目录
相关文章
|
3月前
【力扣】-- 移除链表元素
【力扣】-- 移除链表元素
42 1
|
3月前
Leetcode第21题(合并两个有序链表)
这篇文章介绍了如何使用非递归和递归方法解决LeetCode第21题,即合并两个有序链表的问题。
56 0
Leetcode第21题(合并两个有序链表)
|
3月前
【LeetCode 27】347.前k个高频元素
【LeetCode 27】347.前k个高频元素
43 0
|
3月前
LeetCode第二十四题(两两交换链表中的节点)
这篇文章介绍了LeetCode第24题的解法,即如何通过使用三个指针(preNode, curNode, curNextNode)来两两交换链表中的节点,并提供了详细的代码实现。
32 0
LeetCode第二十四题(两两交换链表中的节点)
|
3月前
Leetcode第十九题(删除链表的倒数第N个节点)
LeetCode第19题要求删除链表的倒数第N个节点,可以通过快慢指针法在一次遍历中实现。
49 0
Leetcode第十九题(删除链表的倒数第N个节点)
|
3月前
【LeetCode 46】450.删除二叉搜索树的节点
【LeetCode 46】450.删除二叉搜索树的节点
24 0
|
3月前
|
索引
力扣(LeetCode)数据结构练习题(3)------链表
力扣(LeetCode)数据结构练习题(3)------链表
109 0
|
3月前
【LeetCode 10】142. 环形链表 II
【LeetCode 10】142. 环形链表 II
26 0
|
3月前
【LeetCode 09】19 删除链表的倒数第 N 个结点
【LeetCode 09】19 删除链表的倒数第 N 个结点
21 0
|
3月前
【LeetCode 08】206 反转链表
【LeetCode 08】206 反转链表
16 0