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.
:::
思路
直接遍历,当遇到当前节点的下一个节点值为目标值,当前节点直接指向下下个结点即可。
但为了防止第一个节点就是要删除的节点,我们需要先构造一个空节点连接到链表上。
var deleteNode = function(head, val) {if (!head) return head;let pre = new ListNode(null);pre.next = head;let cur = pre;while (cur.next) {if (cur.next.val === val) {cur.next = cur.next.next;return pre.next}cur = cur.next;}return null;};
19. 删除链表的倒数第 N 个结点
给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点。
:::info
示例:
给定一个链表: 1->2->3->4->5, 和 n = 2.
当删除了倒数第二个节点后,链表变为 1->2->3->5.
:::
思路
解法一
遍历一次,记录节点个数count,然后求倒数第n个,即求count - n个,此时可以拿到待删除节点的前一个节点prevNode。
var removeNthFromEnd = function(head, n) {let node = new ListNode(0, head);let cur1 = node;let cur2 = node;let count = 1while (cur1.next) {cur1 = cur1.next;count++}let i = 1;while (cur2.next && i < count - n) {cur2 = cur2.next;i++}cur2.next = cur2.next.next;return node.next;}
解法二
在解法一中,需要计算链表长度。为了解决这个问题,我们可以采用快慢指针。
先让快指针fast走n-1步,然后快慢指针同时走。
当快指针到达链表尾节点时,慢指针指向待删除节点的前一个节点。
var removeNthFromEnd = function(head, n) {let node = new ListNode(0, head);let fast = node, slow = node;let count = 0;while (fast.next && count < n) {fast = fast.next;count++}while (fast.next) {fast = fast.next;slow = slow.next;}slow.next = slow.next.next;return node.next};
