力扣第二十五题——K个一组反转链表

内容介绍

给你链表的头节点 head ,每 k 个节点一组进行翻转,请你返回修改后的链表。

k 是一个正整数,它的值小于或等于链表的长度。如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。

你不能只是单纯的改变节点内部的值,而是需要实际进行节点交换。

示例 1:

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

示例 2:

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

提示:
  • 链表中的节点数目为 n
  • 1 <= k <= n <= 5000
  • 0 <= Node.val <= 1000

进阶:你可以设计一个只用 O(1) 额外内存空间的算法解决此问题吗?

完整代码

 class Solution {public ListNode reverseKGroup(ListNode head, int k) {ListNode hair = new ListNode(0);hair.next = head;ListNode pre = hair;while (head != null) {ListNode tail = pre;// 查看剩余部分长度是否大于等于 kfor (int i = 0; i < k; ++i) {tail = tail.next;if (tail == null) {return hair.next;}}ListNode nex = tail.next;ListNode[] reverse = myReverse(head, tail);head = reverse[0];tail = reverse[1];// 把子链表重新接回原链表pre.next = head;tail.next = nex;pre = tail;head = tail.next;}return hair.next;}public ListNode[] myReverse(ListNode head, ListNode tail) {ListNode prev = tail.next;ListNode p = head;while (prev != tail) {ListNode nex = p.next;p.next = prev;prev = p;p = nex;}return new ListNode[]{tail, head};}
}

思路详解

一、解决思路

  1. 辅助头节点:创建一个辅助头节点hair,其下一个节点指向原链表的头节点head。这样做的好处是在翻转链表的过程中,可以简化边界条件的处理。
  2. 分组检查:使用一个循环来检查链表中剩余的节点是否至少有 k 个,以决定是否进行翻转。
  3. 翻转链表:对于每一组 k 个节点,使用一个辅助函数myReverse来进行翻转。
  4. 重新连接:翻转后,需要将翻转的子链表重新连接到原链表中。

二、详细步骤

  1. 初始化辅助头节点

    ListNode hair = new ListNode(0);
    hair.next = head;
    ListNode pre = hair;
    

    这里pre节点用于在翻转后重新连接链表。

  2. 遍历链表

    while (head != null) {
    

    head不为空时,继续处理链表。

  3. 检查剩余节点数量

    ListNode tail = pre;
    for (int i = 0; i < k; ++i) {tail = tail.next;if (tail == null) {return hair.next;}
    }
    

    通过一个循环,检查从当前pre节点开始的 k 个节点是否存在。如果不足 k 个,则直接返回辅助头节点的下一个节点。

  4. 记录翻转后的子链表

    ListNode nex = tail.next;
    ListNode[] reverse = myReverse(head, tail);
    head = reverse[0];
    tail = reverse[1];
    

    myReverse函数翻转从headtail的子链表,并返回翻转后的头尾节点。

  5. 重新连接链表

    pre.next = head;
    tail.next = nex;
    pre = tail;
    head = tail.next;
    

    将翻转后的子链表连接回原链表,并更新prehead指针。

  6. 翻转函数

    public ListNode[] myReverse(ListNode head, ListNode tail) {ListNode prev = tail.next;ListNode p = head;while (prev != tail) {ListNode nex = p.next;p.next = prev;prev = p;p = nex;}return new ListNode[]{tail, head};
    }
    

    该函数通过迭代的方式翻转链表,直到p指向tail

四、返回结果

return hair.next;

最终返回辅助头节点的下一个节点,即翻转后的链表头节点。

通过以上步骤,我们可以实现每 k 个一组翻转链表的功能。该算法的时间复杂度为 O(n),空间复杂度为 O(1),其中 n 是链表中的节点数量。

知识点精炼

一、链表基本概念

  1. 链表是由一系列节点组成的数据结构,每个节点包含数据域和指针域。
  2. 链表的第一个节点称为头节点,最后一个节点的指针域为空。

二、K个一组翻转链表核心知识点

  1. 辅助头节点:引入辅助头节点简化边界条件处理,便于统一操作。
  2. 分组检查:通过循环检查链表剩余节点是否达到 k 个,以决定是否进行翻转。
  3. 链表翻转:使用迭代方法实现链表翻转,保持翻转过程中节点间的连接。
  4. 重新连接:翻转后的子链表需要重新连接到原链表中,保持链表的完整性。

三、关键步骤

  1. 初始化:创建辅助头节点,初始化前驱节点pre
  2. 遍历与检查:遍历链表,检查每组是否有 k 个节点。
  3. 翻转操作:对每组 k 个节点进行翻转,记录翻转后的头尾节点。
  4. 连接链表:将翻转后的子链表连接回原链表,并更新前驱节点和当前节点。

四、注意事项

  1. 边界条件:确保在节点数量不足 k 个时,不进行翻转操作。
  2. 指针更新:在翻转和连接操作中,正确更新指针,避免链表断裂。
  3. 辅助函数:编写清晰的辅助函数,简化主函数逻辑。

五、实际应用

  1. 链表操作:掌握 K 个一组翻转链表,提高链表操作能力。
  2. 算法思维:通过递归和迭代结合的方式,培养灵活的算法思维。

 

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

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

相关文章

自定义IPython启动:打造个性化的交互式编程环境

自定义IPython启动&#xff1a;打造个性化的交互式编程环境 IPython&#xff0c;一个强大的交互式Python解释器&#xff0c;提供了丰富的定制选项&#xff0c;允许用户根据个人或团队的需求定制其行为和外观。设置自定义的启动命令是IPython定制功能的一部分&#xff0c;它可以…

1万6千多传统文化之古代地名大全ACCESS\EXCEL数据库

继续采集传统文化数据&#xff0c;之前采集过《传统文化之非物质文化大全ACCESS数据库》、《名人大全古今人物ACCESS数据库》、《传统文化之民俗文化大全ACCESS数据库》&#xff0c;今天的是古代地名查询&#xff1a; 截图下方有显示“共有记录数”&#xff0c;截图包含了表的所…

探索WebKit的CSS盒模型:深入理解Web布局的基石

探索WebKit的CSS盒模型&#xff1a;深入理解Web布局的基石 在Web开发的世界中&#xff0c;CSS盒模型&#xff08;Box Model&#xff09;是构建网页布局的核心原理。WebKit&#xff0c;作为Safari浏览器的渲染引擎&#xff0c;对CSS盒模型有着深入而精确的支持。本文将带你深入…

AtCoder Beginner Contest 363

A - Piling Up 题意 不同的分数段有不同的^数量&#xff0c;Takahashi想要使得他的^数量增加&#xff0c;问他所需要的最少分数增幅。 思路 我们只需要找到下一阶段的下限。 a / 100 是本阶段 1 变成下一阶段&#xff0c;再 * 100变成下限&#xff0c;再与原来的相减即可…

IP协议和路由转发

文章目录 IP协议IP报头网段划分特殊的IP私有IP和公有IP IP分片 路由 IP协议 IP协议提供了一种能力&#xff0c;将数据报从A主机送到B主机&#xff0c;TCP可以保证可靠性&#xff0c;所以TCP/IP协议可以将数据可靠的从A主机送到B主机。 IP报头 4位版本号(version): 指定IP协议…

Unity Shader - 2024 工具篇

目录 IDE 工具建议 IDE工具 Sublime 3 大势所趋&#xff0c;但是Sublime 使用插件还是相当的不习惯 代码跳转 Go to definite IDE 工具建议 () what is the best ide for coding shaderlab - #4 by DaveAstator - Unity Engine - Unity Discussions​​​​​​​I IDE工…

java中介模式

中介者&#xff08;Mediator&#xff09;模式是一种行为设计模式&#xff0c;它定义了一个对象来封装一组对象之间的交互。中介者对象负责控制和协调这些对象之间的交互&#xff0c;使得这些对象不需要显式地相互引用&#xff0c;从而降低它们之间的耦合。 以下是一个简单的中…

修复SteamUI.dll加载失败的指南,快速修复failed to load steamui.dll

在使用Steam平台进行游戏下载、安装和运行时&#xff0c;可能会遇到一些系统错误&#xff0c;比如“failed to load steamui.dll”。这个错误通常意味着Steam的用户界面库文件steamui.dll出现了问题。本文将详细介绍steamui.dll文件的相关信息以及如何修复这一问题。 一.什么是…

android audio 相机按键音:(二)加载与修改

相机按键音资源&#xff0c;加载文件路径&#xff1a; frameworks/av/services/camera/libcameraservice/CameraService.cpp 按键音&#xff0c;加载函数&#xff1a; void CameraService::loadSoundLocked(sound_kind kind) { ATRACE_CALL(); LOG1("Cam…

MySQL数据库基本安装与部署

目录 概念 数据库的基本概念 关系型数据库 非关系型数据库 MySQL 商业版与社区版 示例 初始化MySQL 添加系统服务 概念 数据库的基本概念 数据&#xff08;Data&#xff09; 描述事物的符号记录包括数字、文字、图形、图像、声音、档案记录等以“记录”形式按统一的…

邮件安全篇:如何防止邮件泄密?

本文主要讨论组织内部用户违反保密规定通过邮件泄密的场景。其他场景导致邮箱泄密的问题&#xff08;如账号被盗、邮件系统存在安全漏洞等&#xff09;不在本文的讨论范围。本文主要从邮件系架构设计、邮件数据防泄漏系统、建立健全规章制度、安全意识培训等方面分别探讨。 1. …

Java响应式编程库Reactor的介绍和基本使用

关于响应式编程的概念和介绍可以参考: 响应式编程(Reactive Programming)是什么? Java语言中,RxJava和Reactor是实现响应式编程的两个最流行的库,因为 Spring 5 及更高版本中,Reactor 是 Spring WebFlux 的底层实现框架,用于构建响应式 Web 应用,所以相比而言,Reactor…

充电桩浪涌保护方案—保障充电设施安全稳定运行的关键

在当今新能源汽车蓬勃发展的时代&#xff0c;充电桩作为电动汽车的“加油站”&#xff0c;其重要性不言而喻。然而&#xff0c;由于其复杂的电气环境和暴露于户外的特点&#xff0c;充电桩容易受到浪涌的影响。浪涌可能来自雷电、电网故障、大功率设备的启停等&#xff0c;对充…

JS+H5可视化广度优先算法

源码在效果图后面 可标记 障碍 起始点 终点 点击寻路按钮后&#xff0c;表格上会自动出现一条蓝色最佳路径&#xff08;加了一格一格显示的动画&#xff09; 以下是效果图 橙色起点 绿色终点 红色障碍物 以下是寻路结果 源代码 <!DOCTYPE html> <html lang&quo…

html+css前端作业 王者荣耀官网6个页面无js

htmlcss前端作业 王者荣耀官网6个页面无js 下载地址 https://download.csdn.net/download/qq_42431718/89571150 目录1 目录2 项目视频 王者荣耀6个页面&#xff08;无js&#xff09; 页面1 页面2 页面3 页面4 页面5 页面6

Camtasia2024官方下载安装激活教程+免费永久许可证秘钥激活码

在这个数字化内容层出不穷的时代&#xff0c;视频的力量日益凸显。无论是用于教育教学、企业培训、内容创作&#xff0c;还是记录生活中的精彩瞬间&#xff0c;一款强大且易用的视频编辑软件都至关重要。今天&#xff0c;让我们一同走进 Camtasia 2024 的精彩世界&#xff0c;探…

分类模型-逻辑回归和Fisher线性判别分析★★★★

该博客为个人学习清风建模的学习笔记&#xff0c;部分课程可以在B站&#xff1a;【强烈推荐】清风&#xff1a;数学建模算法、编程和写作培训的视频课程以及Matlab等软件教学_哔哩哔哩_bilibili 目录 1理论 1.1逻辑回归模型 1.2线性概率模型 1.3线性判别分析 1.4两点分布…

github的Codespaces是什么

目录 github的Codespaces是什么 一、定义与功能 二、特点与优势 三、工作原理 四、使用场景与限制 github的Codespaces是什么 GitHub的Codespaces是一个基于云的即时开发环境,它利用容器技术为开发者提供一个完全配置好的开发环境,以便他们能够直接在浏览器或通过Visua…

轻松上手微调大语言模型——QLORA篇

轻松上手微调大语言模型——QLORA篇 前言 本篇Blog是代码实战篇&#xff0c;着重于新手如何在消费级GPU上快速跑通代码&#xff0c;打造属于自己的AI大语言模型。涉及到相关技术的具体原理我会在以后的更新中一一讲解。 在本篇中我对llama-2进行微调作为示例。 先决条件 安…

vue3学习记录1:emit的写法

emit是用于child组件向parent组件通信的工具&#xff0c;因为vue3的script可以设置为setup&#xff0c;写法同vue2有较大区别。 一、script setup - 直接写 <script lang"ts" setup>const emit defineEmits([close]);function handleClose() {emit(close);}…