Java每日一题——>19. 删除链表的倒数第 N 个结点

这是LeetCode上的 [19,删除链表的倒数第 N 个结点],难度为 [中等]

题目
给你一个链表,删除链表的倒数第 n 个结点,并且返回链表的头结点。

在这里插入图片描述示例1

输入:head = [1,2,3,4,5], n = 2
输出:[1,2,3,5]

示例2

输入:head = [1], n = 1
输出:[]

示例2

输入:head = [1,2], n = 1
输出:[1]

题解(双指针)
思路分析

删除链表的倒数第n个结点,我们只需要找到倒数第n+1个结点(倒数第n个结点的前一个结点),然后让倒数第n+1个结点指向倒数第n-1个结点(倒数第n个结点的下一个结点),就可以实现删除链表的倒数第n个结点

步骤

  1. 定义两个指针front,back指向头结点

  2. 第1个指针front从链表的头结点开始遍历n步,第2个指针back保持不动;

  3. 从第n+1步开始指针back也从链表的头结点开始和指针front以相同的速度遍历。

  4. 当指针p1指向链表的尾节点时指针p2正好指向倒数第n+1个节点(两个指针的距离始终保持为n)

  5. 把倒数第n+1个节点的next指针指向倒数第n-1个节点

注意:由于head是链表的第一个结点(带数据),为了减少判断head是否为null逻辑,以及防止删除头结点出现空指针异常等情况,需要提供一个哨兵结点(虚拟结点不带数据)作为头结点

代码实现

解类如下,类里包含两个方法,一个是遍历链表的,一个是删除链表的倒数第 N 个结点的

public class Solution {

    public static ListNode removeNthFromEnd(ListNode head, int n) {
        // 定义一个哨兵结点作为虚拟头结点,这样可以减少判断头结点是否为null和删除头结点逻辑判断
        ListNode<Integer> dummy = new ListNode<>(0);
        // 让哨兵结点指向头结点
        dummy.next = head;
        // 定义两个结点初始化为哨兵结点
        ListNode front = dummy;
        ListNode back = dummy;
        // front遍历n步
        for (int i = 0; i < n; i++) {
            front = front.next;
        }
         /* front,back以相同的速度遍历,front接上次位置遍历,back从哨兵遍历
          循环终止条件为front到达尾结点时*/
        while (front.next != null) {
            front = front.next;
            back = back.next;
        }
        // 此时back的下一个结点就是要删除的结点,让back指向下一个结点的下一个结点
        back.next = back.next.next;
        return dummy.next;
    }

    public static void printList(ListNode head) {
        System.out.print("[");
        while (head != null) {
            if (head.next == null) {
                System.out.print(head.val);
            } else {
                System.out.print(head.val + ",");
            }
            head = head.next;
        }
        System.out.print("]");
    }
}

结点类

public class ListNode<T> {

    T val;
    ListNode next;

    public ListNode() {

    }

    public ListNode(T val) {
        this.val = val;
    }

    public ListNode(T val, ListNode next) {
        this.next = next;
    }
}

测试类

public class Test {

    @org.junit.Test
    public void test1() {
        ListNode<Integer> head = new ListNode<>(1);
        ListNode<Integer> listNode = new ListNode<>(2);
        ListNode<Integer> listNode1 = new ListNode<>(3);
        ListNode<Integer> listNode2 = new ListNode<>(4);
        ListNode<Integer> listNode3 = new ListNode<>(5);
        head.next = listNode;
        listNode.next = listNode1;
        listNode1.next = listNode2;
        listNode2.next = listNode3;
        System.out.print("输入:");
        Solution.printList(head);
        System.out.println(" 2");
        ListNode head2 = Solution.removeNthFromEnd(head, 2);
        System.out.print("输出:");
        Solution.printList(head2);
    }

    @org.junit.Test
    public void test2() {
        ListNode<Integer> head = new ListNode<>(1);
        System.out.print("输入:");
        Solution.printList(head);
        System.out.println(" 1");
        ListNode head2 = Solution.removeNthFromEnd(head, 1);
        System.out.print("输出:");
        Solution.printList(head2);
    }

    @org.junit.Test
    public void test3() {
        ListNode<Integer> head = new ListNode<>(1);
        ListNode<Integer> listNode = new ListNode<>(2);
        head.next = listNode;
        System.out.print("输入:");
        Solution.printList(head);
        System.out.println(" 2");
        ListNode head2 = Solution.removeNthFromEnd(head, 1);
        System.out.print("输出:");
        Solution.printList(head2);
    }
}

测试结果
在这里插入图片描述在这里插入图片描述在这里插入图片描述

复杂度分析
在这里插入图片描述

假设链表长度为n

时间复杂度:需要遍历链表,故时间复杂度为O(n)
空间复杂度:只声明几个固定的结点,故空间复杂度为O(1);

  • 15
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

stormzhuo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值