LeetCode刷题--- 61. 旋转链表(快慢指针+闭合为环)

news2025/6/30 7:52:06

请添加图片描述

  • 💌 所属专栏:【LeetCode题解(持续更新中)】

  • 😀 作  者:我是夜阑的狗🐶

  • 🚀 个人简介:一个正在努力学技术的码仔,专注基础和实战分享 ,欢迎咨询!

  • 💖 欢迎大家:这里是CSDN,我总结知识的地方,喜欢的话请三连,有问题请私信 😘 😘 😘

您的点赞、关注、收藏、评论,是对我最大的激励和支持!!!🤩 🤩 🤩

文章目录

  • 前言
  • 一、编程题: 61. 旋转链表(快慢指针+闭合为环)
      • 1.题目描述
      • 2.示例1:
      • 3.示例2:
      • 4.提示:
  • 二、解题思路
      • 1. 方法1(快慢指针)
        • 思路:
        • 复杂度分析:
        • 算法图解
      • 2. 方法二:闭合为环
        • 思路:
        • 复杂度分析:
        • 算法图解
  • 三、代码实现
  • 总结


前言

  大家好,又见面了,我是夜阑的狗,本文是专栏【LeetCode题解】专栏的第20篇文章,主要讲解的是LeetCode 61. 旋转链表(快慢指针+闭合为环))。
  专栏地址:【LeetCode题解(持续更新中)】, 此专栏是我是夜阑的狗对LeetCode问题的讲解,希望能够加深自己的印象,以及帮助到其他的小伙伴😉😉。
  如果文章有什么需要改进的地方还请大佬不吝赐教👏👏。


一、编程题: 61. 旋转链表(快慢指针+闭合为环)

1.题目描述

  给你一个链表的头节点 head ,旋转链表,将链表每个节点向右移动 k 个位置。LeetCode题目链接。

2.示例1:

在这里插入图片描述

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

3.示例2:

在这里插入图片描述

输入:head = [0,1,2], k = 4
输出:[2,0,1]

4.提示:

  • 链表中节点的数目在范围 [0, 500] 内
  • -100 <= Node.val <= 100
  • 0 <= k <= 2 * 109

二、解题思路

1. 方法1(快慢指针)

  这里使用快慢指针跟滑动窗口的思路是一样的,都是先找到倒数k+1个节点,然后在一起移动;

思路:

  • Step 1.创建快慢指针,遍历链表找到快指针位置(倒数第k+1个节点的);
  • Step 2.同时移动快慢指针直到快指针fast == null为止,此时慢指针指向新链表的尾节点(这里还要一个指针来记录快指针的前驱节点,这个要看移动情况来定用不用);
  • Step 3.对原链表进行拆分并返回新链表头节点即可;

看不懂解释的话,直接看算法图解比较容易理解点

复杂度分析:

时间复杂度:O(n)。
空间复杂度:O(1)。

算法图解

  找到快指针之后就同时移动快慢指针,红色部分代表着快指针,只有fast==null时才要记录其前驱节点fast_prev,为了后续链表拼接操作;(注:本人不会做成流程动画,希望会的朋友可以私信我指点一二,说个软件名字也可以,谢谢 🤩 🤩 🤩

请添加图片描述

2. 方法二:闭合为环

  假设链表的长度为n,可以发现这么规律,当向右移动次数k ≥ n时,只要移动k mod n次即可。而且每次n(或者n的倍数)次移动都是链表的原状态,所以推导出新链表的头节点位置为原链表的第n - (k mod n)个节点(从0开始计数);

思路:

  • Step 1.首先将链表够成环,遍历链表找到链表的尾节点,并计算出链表的长度list_length,然后将其与头节点相连(这里有个坑点,就是找链表尾节点的时候,循环终止条件是last.next != null,这里的计算出来的链表长度都要+1);
  • Step 2.由公式n-(k%n)可计算出新链表头节点在原链表的位置,根据该位置可以找到其前驱节点(也就是新链表的尾节点),当k为n的倍数时则不进行处理;
  • Step 3.找到新链表的尾节点之后,对链表进行拆环并返回新链表头节点即可;

看不懂解释的话,直接看算法图解比较容易理解点

复杂度分析:

时间复杂度:O(n),最坏情况下,我们需要遍历该链表两次。
空间复杂度:O(1)。我们只需要常数的空间存储若干变量。

算法图解

  红色部分代表当前遍历的节点,注意这里Kth的位置是从0开始的,Kth=3对应的位置4;(注:本人不会做成流程动画,希望会的朋友可以私信我指点一二,说个软件名字也可以,谢谢

请添加图片描述


三、代码实现

每个代码块都写了注释,方便理解,代码还可以改进;

方法一:快慢指针

class Solution {
    public ListNode rotateRight(ListNode head, int k) {
        // 预处理阶段
        if(head == null || head.next == null || k == 0) return head;
        ListNode temp = new ListNode(0, head);
        // 创建快慢指针
        ListNode fast = head, slow = temp, fast_prev = head; 
        int listlength = 0; // 链表长度

        // 找到快指针的位置, 新链表的尾节点
        for(int i = 0; i < k; i++){
            // 链表遍历结束时,k值还没有结束
            if(fast == null){
                fast = head.next;
                int templ = (k-1)%listlength;
                // System.out.println("templ = " + templ);
                for(int j = 0; j < templ; j++){
                    fast = fast.next;
                }
                // 找到快指针结束循环
                break;
            }
            fast = fast.next;
            listlength++;
        }
        if(fast == null) return head;
        // System.out.println("fast.val = " + fast.val);
        // 同时移动快慢指针
        while(fast != null){
            if(fast.next == null) fast_prev = fast;
            fast = fast.next;
            slow = slow.next;
        }

        // System.out.println("slow.val = " + slow.val);
        fast_prev.next = head;
        head = slow.next;
        slow.next = null;

        return head;
    }
}

提交结果:

在这里插入图片描述

方法二:闭合为环

class Solution {
    public ListNode rotateRight(ListNode head, int k) {
        // 预处理阶段
        if(k == 0 || head == null || head.next == null){
            return head;
        }
        
        int list_length = 1, Kth = 0;
        ListNode last = head, first;
        // 找到链表尾节点,并计算链表长度
        while(last.next != null){
            list_length++;
            last = last.next;
        }

        // (n-1)-(k%n)
        Kth = list_length - (k % list_length);
        if(Kth == list_length) return head;
        // 构建环
        last.next = head;
        // 根据公式计算出新链表的头节点位置并拆解环
        while(Kth-- > 0) last = last.next;
        first = last.next;
        last.next = null;

        return first;
    }
}

提交结果:

在这里插入图片描述


总结

感谢观看,如果有帮助到你,请给题解点个赞和收藏,让更多的人看到。🌹 🌹 🌹

也欢迎你,关注我。👍 👍 👍

原创不易,还希望各位大佬支持一下,你们的点赞、收藏和留言对我真的很重要!!!💕 💕 💕 最后,本文仍有许多不足之处,欢迎各位认真读完文章的小伙伴们随时私信交流、批评指正!

更多专栏订阅:

  • 😀 【LeetCode题解(持续更新中)】
  • 🚝 【Java Web项目构建过程】
  • 💛 【数字图像处理】
  • 【JavaScript随手笔记】
  • 🤩 【大数据学习笔记(华为云)】
  • 🦄 【程序错误解决方法(建议收藏)】
  • 🚀 【软件安装教程】



订阅更多,你们将会看到更多的优质内容!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/345527.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

cors跨域问题

CORS CORS&#xff0c;全称Cross-Origin Resource Sharing,是一种允许当前域&#xff08;domain&#xff09;的资源&#xff08;比如html/js/web service&#xff09;被其他域&#xff08;domain&#xff09;的脚本请求访问的机制&#xff0c;通常由于同域安全策略&#xff08;…

如何设计一个秒杀系统

秒杀系统要如何设计&#xff1f; 前言 高并发下如何设计秒杀系统&#xff1f;这是一个高频面试题。这个问题看似简单&#xff0c;但是里面的水很深&#xff0c;它考查的是高并发场景下&#xff0c;从前端到后端多方面的知识。 秒杀一般出现在商城的促销活动中&#xff0c;指定…

Cosmos 基础 -- Ignite CLI(二)Module basics: Blog

一、快速入门 Ignite CLI version: v0.26.1 在本教程中&#xff0c;我们将使用一个模块创建一个区块链&#xff0c;该模块允许我们从区块链中写入和读取数据。这个模块将实现创建和阅读博客文章的功能&#xff0c;类似于博客应用程序。最终用户将能够提交新的博客文章&#x…

计算机网络学习笔记(一)

网络是由若干接点和连接这些结点的链路组成。 多个网络通过路由器互联起来构成覆盖范围更大的互联网。 普通用户通过ISP接入因特网。 基于ISP的三层结构因特网 相隔较远的两台主机间通信可能需要经过多个ISP。 有电路交换&#xff0c;报文交换&#xff0c;分组交换三种交换方…

【并发编程】LockSupport源码详解

目录 一、前言 1.1 简介 1.2 为什么说LockSupport是Java并发的基石&#xff1f; 二、LockSupport的用途 2.1 LockSupport的主要方法 2.2 使用案例 2.3 总结 三、LockSupport 源码分析 3.1 学习原理前的前置知识 3.1.1 Unsafe.park()和Unsafe.unpark() 3.1.2wait和notify/notify…

MyEclipse技术全面解析——EJB开发工具介绍(一)

MyEclipse v2022.1.0正式版下载1. MyEclipse EJB开发工具Enterprise Java Beans (EJB) 已经成为实现Java企业业务功能和与数据库资源接口的Java EE 5标准&#xff0c;MyEclipse EJB3工具支持Java EE 5简化的基于注释的POJO编程模型&#xff0c;这些工具使开发人员能够在几分钟内…

微信怎么群发消息给所有人?图文教学,快速弄懂

​微信作为很多小伙伴经常使用的工具&#xff0c;无论是学习、工作还是其他方面都会使用到。有些时候&#xff0c;需要将同一条消息发给通讯录里的大多数人&#xff0c;一条一条的转发太慢了&#xff0c;群发消息给所有人是个不错的办法。微信怎么群发消息给所有人&#xff1f;…

广东省基层就业补贴

基层就业补贴链接&#xff1a;https://www.gdzwfw.gov.cn/portal/v2/guide/11440309MB2D27065K4440511108001 一.申请条件&#xff1a; 1、劳动者到中小微企业、个体工商户、社会组织等就业&#xff0c;或到乡镇&#xff08;街道&#xff09;、村居社会管理和公共服务岗位就业…

spring cloud篇——什么是服务熔断?服务降级?服务限流?spring cloud有什么优势?

文章目录一、spring cloud 有什么优势二、服务熔断2.1、雪崩效应2.2、DubboHystrixCommand三、服务降级四、服务限流4.1、限流算法4.2、应用级限流4.3、池化技术4.4、分布式限流4.5、基于Redis 功能的实现限流4.6、基于令牌桶算法的实现4.6.1 、Java实现一、spring cloud 有什么…

GUI swing和awt

GUI&#xff08;Graphical User Interface&#xff0c;简称 GUI&#xff0c;图形用户界面&#xff09;是指采用图形方式显示的计算机操作用户界面&#xff0c;与早期计算机使用的命令行界面相比&#xff0c;图形界面对于用户来说在视觉上更易于接受。Java GUI主要有两个核心库&…

【计算机网络】传输层TCP协议

文章目录认识TCP协议TCP协议的格式字段的含义序号与确认号六个标志位窗口大小确认应答(ACK)机制超时重传机制连接管理机制三次握手四次挥手滑动窗口流量控制拥塞控制延迟应答捎带应答面向字节流粘包问题TCP异常情况总结认识TCP协议 传输控制协议 &#xff08;TCP&#xff0c;T…

多边形网格算法笔记

本文是处理多边形和网格的各种笔记和算法。 推荐&#xff1a;使用 NSDT场景设计器 快速搭建 3D场景。 1、表面简化 下面描述了一种方法&#xff0c;用于减少构成表面表示的多边形数量&#xff0c;同时仍试图保留表面的基本形式。 如果正在为渲染和/或交互环境寻求性能改进&…

【CS224图机器学习】task1 图机器学习导论

前言&#xff1a;本期学习是由datawhale&#xff08;公众号&#xff09;组织&#xff0c;由子豪兄讲解的202302期CS224图机器学习的学习笔记。本次学习主要针对图机器学习导论做学习总结。1.什么是图机器学习&#xff1f;通过图这种数据结构&#xff0c;对跨模态数据进行整理。…

增减序列(差分)

分析&#xff1a;要想把整个数组变为同一个数&#xff0c;我们可以根据差分的思想来做。 差分定义&#xff1a;b[1]a[1] b[2]a[2]-a[1] ...... b[i]a[i]-a[i-1] 由定义可知&#xff0c;可以把b[2]~b[n]全部变为0&#xff0c;那么整个数组就一样了。现在问题转换为如何用最少的…

Seata-Server分布式事务原理加源码 (八) - Seata-XA模式

Seata-XA模式 Seata 1.2.0 版本重磅发布新的事务模式&#xff1a;XA 模式&#xff0c;实现对 XA 协议的支持。 我们从三个方面来深入分析&#xff1a; XA模式是什么&#xff1f;为什么支持XA&#xff1f;XA模式如何实现的&#xff0c;以及如何使用&#xff1f; XA模式 首先…

shell学习1

目录 一、echo 1.1 echo 1.2 打印彩色文本 1.3 打印彩色背景 二、printf 三、变量和环境变量 3.1 查看某个进程的环境变量 3.2给变量赋值。varvalue 3.3 给环境变量赋值 3.4 获取变量的长度 3.5 识别当前所使用的shell 3.6 检查是否为超级用户 四、数学运算 4.1 …

PHP新特性集合

php8新特性命名参数function foo(string $a, string $b, ?string $c null, ?string $d null) { /* … */ }你可以通过下面的方式传入参数进行调用foo(b: value b, a: value a, d: value d, );联合类型php7class Number {/** var int|float */private $number;/*** param f…

Vue|事件处理

事件处理1. 事件使用1.1 事件绑定1.2 事件参数2. 事件修饰符2.1 阻止默认事件2.2 阻止事件冒泡2.3 事件只允许触发一次2.4 事件捕获2.5 操作当前元素2.6 行为立即执行无需等待回调3. 键盘事件4. 本章小结4.1 事件使用小结4.2 事件修饰符小结4.3 键盘事件小结1. 事件使用 1.1 事…

C++STL剖析(八)—— unordered_set和unordered_multiset的概念和使用

文章目录前言1. unordered_set的介绍和使用&#x1f351; unordered_set的构造&#x1f351; unordered_set的使用&#x1f345; insert&#x1f345; find&#x1f345; erase&#x1f345; size&#x1f345; empty&#x1f345; clear&#x1f345; swap&#x1f345; count…

安全多方计算系列笔记1——前世今生

这一系列笔记参考了绿盟科技研究通讯的安全多方计算文章&#xff0c;及其他。 首先看定义&#xff1a;在不泄露参与方原始输入数据的前提下&#xff0c;允许分布式参与方合作计算任意函数&#xff0c;输出准确的计算结果。 起源 安全多方计算问题及解首先由姚期智&#xff08…