18. 删除链表的节点

给定单向链表的头指针和一个要删除的节点的值,定义一个函数删除该节点。
返回删除后的链表的头节 :::info 示例 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. :::

思路

直接遍历,当遇到当前节点的下一个节点值为目标值,当前节点直接指向下下个结点即可。
但为了防止第一个节点就是要删除的节点,我们需要先构造一个空节点连接到链表上。

  1. var deleteNode = function(head, val) {
  2. if (!head) return head;
  3. let pre = new ListNode(null);
  4. pre.next = head;
  5. let cur = pre;
  6. while (cur.next) {
  7. if (cur.next.val === val) {
  8. cur.next = cur.next.next;
  9. return pre.next
  10. }
  11. cur = cur.next;
  12. }
  13. return null;
  14. };

19. 删除链表的倒数第 N 个结点

给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点。 :::info 示例:
给定一个链表: 1->2->3->4->5, 和 n = 2.
当删除了倒数第二个节点后,链表变为 1->2->3->5. :::

思路

解法一

遍历一次,记录节点个数count,然后求倒数第n个,即求count - n个,此时可以拿到待删除节点的前一个节点prevNode

  1. var removeNthFromEnd = function(head, n) {
  2. let node = new ListNode(0, head);
  3. let cur1 = node;
  4. let cur2 = node;
  5. let count = 1
  6. while (cur1.next) {
  7. cur1 = cur1.next;
  8. count++
  9. }
  10. let i = 1;
  11. while (cur2.next && i < count - n) {
  12. cur2 = cur2.next;
  13. i++
  14. }
  15. cur2.next = cur2.next.next;
  16. return node.next;
  17. }

解法二

在解法一中,需要计算链表长度。为了解决这个问题,我们可以采用快慢指针。
先让快指针fastn-1步,然后快慢指针同时走。
当快指针到达链表尾节点时,慢指针指向待删除节点的前一个节点。
删除链表节点 - 图1

  1. var removeNthFromEnd = function(head, n) {
  2. let node = new ListNode(0, head);
  3. let fast = node, slow = node;
  4. let count = 0;
  5. while (fast.next && count < n) {
  6. fast = fast.next;
  7. count++
  8. }
  9. while (fast.next) {
  10. fast = fast.next;
  11. slow = slow.next;
  12. }
  13. slow.next = slow.next.next;
  14. return node.next
  15. };