算法日常记录

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;
    }
}
相关推荐
wwww.bo3 分钟前
机器学习(决策树)
算法·决策树·机器学习
辞--忧7 分钟前
深入浅出决策树
算法·决策树·机器学习
Y200309168 分钟前
决策树总结
算法·决策树·机器学习
lynn8570_blog8 分钟前
低端设备加载webp ANR
前端·算法
LKAI.29 分钟前
传统方式部署(RuoYi-Cloud)微服务
java·linux·前端·后端·微服务·node.js·ruoyi
HeyZoeHey31 分钟前
Mybatis执行sql流程(一)
java·sql·mybatis
2301_7930868734 分钟前
SpringCloud 07 微服务网关
java·spring cloud·微服务
图灵学术计算机论文辅导1 小时前
论文推荐|迁移学习+多模态特征融合
论文阅读·人工智能·深度学习·计算机网络·算法·计算机视觉·目标跟踪
threejs源码翻译官2 小时前
显微镜图像处理【优化】- 使用图像风格迁移技术放大图像细节
算法
柳贯一(逆流河版)2 小时前
Spring 三级缓存:破解循环依赖的底层密码
java·spring·缓存·bean的循环依赖