删除链表的倒数第N个节点
题目
给定一个链表, 删除链表的倒数第 n
个节点, 并返回修改后的链表的头节点.
提示:
- 链表中结点的数目为
sz
1 <= sz <= 30
0 <= Node.val <= 100
1 <= n <= sz
示例
输入: 给定一个链表 1 -> 2 -> 3 -> 4 -> 5
, n = 2
输出: 1 -> 2 -> 3 -> 5
题解
- JavaScript - 朴素解法
- JavaScript - 双指针解法
- Rust
最朴素的解法是先遍历链表, 获取链表的长度, 从而根据倒数第 n
个节点算出该节点的 index
; 然后使用删除链表节点的方法将该节点删除.
/**
* Definition for singly-linked list.
* function ListNode(val) {
* this.val = val;
* this.next = null;
* }
*/
/**
* @param {ListNode} head
* @param {number} n
* @return {ListNode}
*/
var removeNthFromEnd = function (head, n) {
let curr = head
let total = 0
while (curr !== null) {
total++
curr = curr.next
}
let deletedNodeIndex = total - n
let prev = head
for (let i = 0; i < deletedNodeIndex - 1; i++) {
prev = prev.next
}
if (deletedNodeIndex === 0) {
return head.next
}
prev.next = prev.next.next
return head
}
- 先让快指针走到
n
的位置 - 如果此时快指针走到头了, 说明倒数第
n
个节点就是第一个节点, 直接用head.next
删除头节点 - 否则在
fast
和fast.next
都存在的情况下, 让慢指针和快指针同步向前 - 当快指针到了最后一个节点, 慢指针也就到了倒数第
n
个节点的前一个节点 - 使用
slow.next = slow.next.next
即可删除该节点 - 返回
head
即可
note
由于第二次遍历是第一次的延续(第一次是从链头到 n
, 第二次是从 n
到链尾), 因此其实就是一趟遍历.
/**
* Definition for singly-linked list.
* function ListNode(val) {
* this.val = val;
* this.next = null;
* }
*/
/**
* @param {ListNode} head
* @param {number} n
* @return {ListNode}
*/
var removeNthFromEnd = function (head, n) {
let fast = head
let slow = head
// 先让快指针走到 n 的位置
while (n--) {
fast = fast.next
}
if (!fast) {
// 如果此时快指针走到头了
// 说明倒数第 n 个节点就是第一个节点, 那么删除头节点相当于直接返回 head.next 即可
return head.next
}
// 让慢指针和快指针同步向前
while (!fast && !fast.next) {
fast = fast.next
slow = slow.next
}
// slow.next 就是倒数第 n 个节点, 删除它
slow.next = slow.next.next
return head
}
danger
老子再也不用 Rust 写链表了!
// Definition for singly-linked list.
#[derive(PartialEq, Eq, Clone, Debug)]
pub struct ListNode {
pub val: i32,
pub next: Option<Box<ListNode>>
}
impl ListNode {
#[inline]
fn new(val: i32) -> Self {
ListNode {
next: None,
val
}
}
}
pub fn remove_nth_from_end(head: Option<Box<ListNode>>, n: i32) -> Option<Box<ListNode>> {
let mut dummy = ListNode::new(0);
dummy.next = head;
let mut dummy = Box::new(dummy);
let mut fast = dummy.clone();
let mut slow = dummy.as_mut();
// move fast n forward
for _ in 0..n {
fast = fast.next.unwrap();
}
while fast.next.is_some() {
fast = fast.next.unwrap();
slow = slow.next.as_mut().unwrap();
}
let next = slow.next.as_mut().unwrap();
slow.next = next.next.clone();
dummy.next
}