19.删除链表的倒数第N个节点
给你一个链表,删除链表的倒数第 n
个结点,并且返回链表的头结点。
示例 1:
输入:head = [1,2,3,4,5], n = 2
输出:[1,2,3,5]
示例 2:
输入:head = [1], n = 1
输出:[]
示例 3:
输入:head = [1,2], n = 1
输出:[1]
提示:
- 链表中结点的数目为
sz
1 <= sz <= 30
0 <= Node.val <= 100
1 <= n <= sz
**进阶:**你能尝试使用一趟扫描实现吗?
题解:
本题的解题思路是使用两个指针,在此我定义三个指针分别为front和back,pre。让front先向前移动n格,然后再让front和back同时向前移动,当front遍历到最后一个的时候,back指针指向的元素就是链表中的倒数第n个, pre为back的前一个,利用pre.next = back.next
实现删除逻辑。
但本题存在一个注意点,就是如果只有一个节点,只是删除第一个,那么就需要删除头节点,这就涉及到了头节点的改变,因此还是用到了之前 删除排序链表中的重复元素Ⅱ 中提到的技巧,创建一个dummy
节点用来指向头节点,最后返回dummy.next
。
举个栗子,链表为[1], n等于1。如下:
使用这种方案,只需要遍历依次链表,即可完成删除,代码实现如下:
package com.offer;
import com.offer.leetcode.datastruct.ListNode;
import com.offer.leetcode.datastruct.ListNodeUtils;
public class _19删除链表的倒数第N个节点 {
public static void main(String[] args) {
ListNode head = ListNodeUtils.createList(new int[]{1});
ListNodeUtils.printListNode(head);
head = removeNthFromEnd(head, 1);
ListNodeUtils.printListNode(head);
}
public static ListNode removeNthFromEnd(ListNode head, int n) {
ListNode front = head;
ListNode back = head;
ListNode dummy = new ListNode(0);
dummy.next = head;
ListNode pre = dummy;
// front向前移动n格
while (front != null && n > 0) {
front = front.next;
n--;
}
while (front != null) {
pre = back;
back = back.next;
front = front.next;
}
pre.next = back.next;
// back即为需要删除的节点
return dummy.next;
}
}