算法日常记录

1. 链表

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

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

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

输出:[1,2,3,5]

思路:先让fast跑n步,然后slow和fast再一起跑,fast到达末尾时,slow刚好为倒数第n+1个节点,下面代码使用了虚拟结点,所以跑n+1步。

1 -> 2 -> 3 -> 4 -> 5-> null

删除倒数第二个结点,即4;那么要使得3的指针指向5,即3 -> 5,在原链表上表现为 3.next = 3.next.next;

即删除结点的前一个结点 指向 该结点的后一个结点

采用双指针进行完成:快指针移动到n+1的位置后,慢指针开始移动

自测代码:

java 复制代码
/**
 * 删除链表的倒数第 N 个结点
 * 输入:head = [1,2,3,4,5], n = 2
 * 输出:[1,2,3,5]
 *
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
public class RemoveNthFromEnd {
    public static void main(String[] args) {

        ListNode head = new ListNode(1, new ListNode(2, new ListNode(3, new ListNode(4, new ListNode(5)))));
        ListNode result = removeNthFromEnd(head,2);
        ListNode curr = result;
        while (curr != null) {
            System.out.println(curr.val);
            curr = curr.next;
        }
    }

    /*
     1 -> 2 -> 3 -> 4 -> 5-> null
     删除倒数第二个结点,即4;那么要使得3的指针指向5,即3 -> 5,在原链表上表现为 3.next = 3.next.next;
     即删除结点的前一个结点 指向 该结点的后一个结点
     采用双指针进行完成:快指针移动到n+1的位置后,慢指针开始移动

     */
    private static ListNode removeNthFromEnd(ListNode head, int n) {
        //为避免要删除结点没有前一个结点,使用虚拟结点指向head
        ListNode dummy = new ListNode(0,head);
        ListNode fast = dummy;
        ListNode slow = dummy;
        //fast结点先移动到n节点的后一个结点位置
        for (int i = 0; i <= n; i++) {
            fast = fast.next;
        }
        //然后fast和slow结点一起移动,知道fast为null
        while (fast != null) {
            fast = fast.next;
            slow = slow.next;
        }

        // 此时 slow 的位置就是待删除元素的前一个位置
        if(slow.next != null){
            slow.next = slow.next.next;
        }
        return dummy.next;
    }

力扣代码:

java 复制代码
/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode removeNthFromEnd(ListNode head, int n) {
        //新建一个虚拟头节点指向head
         ListNode dummy = new ListNode(0, head);
         //快慢指针指向虚拟头节点
         ListNode fast = dummy;
         ListNode slow =dummy;
         //快指针先移动n+1个节点,只要快慢指针相差 n 个结点即可
         for(int i=0; i<=n;i++){
            fast = fast.next;
         }
         //此时,快慢指针一起移动,快指针为null时,慢指针位于n的前一个节点
         while(fast != null){
            fast = fast.next;
            slow = slow.next;
         }
         // 此时 slow 的位置就是待删除元素的前一个位置
         if(slow.next != null){
            slow.next = slow.next.next;
         }

         return dummy.next;
    }
}
相关推荐
Hx_Ma165 小时前
SpringMVC框架提供的转发和重定向
java·开发语言·servlet
期待のcode6 小时前
原子操作类LongAdder
java·开发语言
舟舟亢亢6 小时前
Java集合笔记总结
java·笔记
A_nanda7 小时前
c# MOdbus rto读写串口,如何不相互影响
算法·c#·多线程
小酒窝.7 小时前
【多线程】多线程打印ABC
java
乡野码圣7 小时前
【RK3588 Android12】RCU机制
java·jvm·数据库
JAVA+C语言7 小时前
如何优化 Java 多主机通信的性能?
java·开发语言·php
编程彩机8 小时前
互联网大厂Java面试:从分布式架构到大数据场景解析
java·大数据·微服务·spark·kafka·分布式事务·分布式架构
代码雕刻家8 小时前
2.4.蓝桥杯-分巧克力
算法·蓝桥杯
Ulyanov9 小时前
顶层设计——单脉冲雷达仿真器的灵魂蓝图
python·算法·pyside·仿真系统·单脉冲