【LeetCode】148. 排序链表
排序链表
题目描述:
给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表 。
示例 1:
输入:head = [4,2,1,3] 输出:[1,2,3,4]
示例 2:
输入:head = [-1,5,3,4,0] 输出:[-1,0,3,4,5]
示例 3:
输入:head = [] 输出:[]
提示:
- 链表中节点的数目在范围
[0, 5 * 104]内 -105 <= Node.val <= 105
进阶:你可以在 O(n log n) 时间复杂度和常数级空间复杂度下,对链表进行排序吗?
思路分析:
使用归并排序
大体实现:
-
基本情况处理:首先检查链表是否为空或只有一个节点。如果是,那么它已经是有序的,直接返回该链表。
-
分割链表:使用快慢指针技术找到链表的中点,并将链表从中点处分割成两个子链表。快指针
fast每次移动两步,慢指针slow每次移动一步,同时用一个prevPtr指针来记录slow的前一个节点,以便在找到中点后将链表分割。 -
递归排序:对分割后的两个子链表递归地调用
sortList方法进行排序。由于递归的性质,这两个子链表最终都会被分割成更小的子链表,直到每个子链表只包含一个节点(或为空),然后这些子链表会被合并成有序链表。 -
合并链表:使用
merge方法将两个已排序的子链表合并成一个有序链表。合并过程中,通过比较两个链表头节点的值,将较小的节点连接到结果链表的末尾,并移动相应的指针。当其中一个链表遍历完时,将另一个链表的剩余部分直接连接到结果链表的末尾。
细节解释
-
快慢指针:快指针
fast每次移动两步,慢指针slow每次移动一步。当快指针到达链表末尾时,慢指针正好位于链表的中点(或中点的前一个节点,如果链表长度为奇数)。prevPtr用于记录slow的前一个节点,以便在找到中点后将链表分割。 -
链表分割:通过将
prevPtr.next设置为null,可以将链表从中点处分割成两个子链表。prevPtr是slow的前一个节点,因此prevPtr.next = null会将slow及其后面的节点与前面的节点断开连接。(在使用快慢指针找到中点并分割链表时,需要确保链表长度至少为2,否则fast.next.next可能会引发NullPointerException。但在这个实现中,由于首先检查了链表是否为空或只有一个节点,所以这种情况不会发生。) -
递归调用:对分割后的两个子链表(
head和slow)public class Solution { // 主函数,用于对链表进行排序 public ListNode sortList(ListNode head) { // 如果链表为空或只有一个节点,则它已经是有序的,直接返回 if (head == null || head.next == null) { return head; } // 使用快慢指针找到链表的中点 ListNode slow = head; // 慢指针,每次移动一步 ListNode fast = head; // 快指针,每次移动两步 ListNode prevPtr = null; // 用于记录slow的前一个节点,以便分割链表 while (fast != null && fast.next != null) { prevPtr = slow; slow = slow.next; fast = fast.next.next; } // 分割链表:将链表从中间断开,prevPtr.next原本指向slow,现在设置为null prevPtr.next = null; // 递归地对左右两部分进行排序 // 注意:这里的left是原链表的左半部分,而right是原链表的右半部分(从slow开始) ListNode left = sortList(head); // 对左半部分排序 ListNode right = sortList(slow); // 对右半部分排序 // 合并两个有序链表 // 返回合并后的链表头节点 return merge(left, right); } // 合并两个有序链表的辅助函数 private ListNode merge(ListNode l1, ListNode l2) { // 创建一个哑节点,方便处理链表头部 ListNode dummy = new ListNode(0); ListNode tail = dummy; // tail用于遍历并连接合并后的链表 // 遍历两个链表,按序连接节点 while (l1 != null && l2 != null) { if (l1.val < l2.val) { tail.next = l1; l1 = l1.next; } else { tail.next = l2; l2 = l2.next; } tail = tail.next; // 移动tail到当前连接的节点 } // 如果l1或l2还有剩余节点,直接将剩余部分连接到tail后面 if (l1 != null) { tail.next = l1; } if (l2 != null) { tail.next = l2; } // 返回合并后的链表(跳过哑节点) return dummy.next; } }分别递归调用
sortList方法进行排序。注意,这里的head是原链表的头节点,而slow是分割后右子链表的头节点。 -
合并链表:使用
merge方法将两个已排序的子链表合并成一个有序链表。合并过程中,通过比较两个链表头节点的值,将较小的节点连接到结果链表的末尾,并移动相应的指针。当其中一个链表遍历完时,另一个链表的剩余部分(如果有的话)将直接连接到结果链表的末尾。
代码实现:
public class Solution { // 主函数,用于对链表进行排序 public ListNode sortList(ListNode head) { // 如果链表为空或只有一个节点,则它已经是有序的,直接返回 if (head == null || head.next == null) { return head; } // 使用快慢指针找到链表的中点 ListNode slow = head; // 慢指针,每次移动一步 ListNode fast = head; // 快指针,每次移动两步 ListNode prevPtr = null; // 用于记录slow的前一个节点,以便分割链表 while (fast != null && fast.next != null) { prevPtr = slow; slow = slow.next; fast = fast.next.next; } // 分割链表:将链表从中间断开,prevPtr.next原本指向slow,现在设置为null prevPtr.next = null; // 递归地对左右两部分进行排序 // 注意:这里的left是原链表的左半部分,而right是原链表的右半部分(从slow开始) ListNode left = sortList(head); // 对左半部分排序 ListNode right = sortList(slow); // 对右半部分排序 // 合并两个有序链表 // 返回合并后的链表头节点 return merge(left, right); } // 合并两个有序链表的辅助函数 private ListNode merge(ListNode l1, ListNode l2) { // 创建一个哑节点,方便处理链表头部 ListNode dummy = new ListNode(0); ListNode tail = dummy; // tail用于遍历并连接合并后的链表 // 遍历两个链表,按序连接节点 while (l1 != null && l2 != null) { if (l1.val < l2.val) { tail.next = l1; l1 = l1.next; } else { tail.next = l2; l2 = l2.next; } tail = tail.next; // 移动tail到当前连接的节点 } // 如果l1或l2还有剩余节点,直接将剩余部分连接到tail后面 if (l1 != null) { tail.next = l1; } if (l2 != null) { tail.next = l2; } // 返回合并后的链表(跳过哑节点) return dummy.next; }
}相关文章:
【LeetCode】148. 排序链表
排序链表 题目描述: 给你链表的头结点 head ,请将其按 升序 排列并返回 排序后的链表 。 示例 1: 输入:head [4,2,1,3] 输出:[1,2,3,4]示例 2: 输入:head [-1,5,3,4,0] 输出:…...
阿里云-java调用短信服务,第三方接口的开启(傻瓜式教程)
第一步:在浏览器中,搜索阿里云 第二步:打开aly的主页 第三步:在最上方的导航栏中,找到云市场,注意不要点击,会自动有触发悬浮框出现,在悬浮框中找到 短信 第四步:点击 短…...
以node / link文件表征的道路网络-----基于南京公路公开数据做路径规划(下)------dijkstra算法的一些简单花样
在不改变dijkstra算法本身的情况下,完全可以从数据源的角度出发,解决我们的一些简单需求: 比较初级且粗暴的玩法,可以是强行赋予一些link极端的路段长度。 对于我们坚决不希望车辆行驶的道路、禁行区、或是危险区,就…...
计算机操作员中级理论知识试题
计算机操作员中级理论知识试题 一、单项选择题 在ASCII编码中,无法显示或打印的字符是()。 A.字符$,%,# B.运算符号*,.,/ C.空格 D.ASCII编码值在0-30间的控制符号将十进制数31.625转换成十六进制数是() A.115.10 B.If.a C.37.5 D.If.10在计算机中,同统一指挥和控制计…...
Redis主从同步配置
1: 安装Redis 参考 linux ubuntu安装redis_ubuntu离线安装redis7.2.5-CSDN博客 2:创建目录 到达redis 根目录 cd /usr/redis/# 创建主从工作目录 mkdir -p replication/6379 # master 节点 mkdir -p replication/6378 # 从节点 mkdir -p replication/6377 # 从节点…...
输出重定向
输出重定向是指将程序的输出(标准输出、错误输出等)重定向到指定的位置,而不是默认的输出设备(通常是终端/控制台)。在 Unix/Linux 系统中,输出重定向通过使用符号 >、>>、2> 等来实现。 常见…...
ubuntu20.04挂载机械硬盘
环境说明 1.基于清华源地址下载的ubuntu20.04制作的系统盘,然后安装在PC上(固态硬盘) 2.机械硬盘无法看见 目的 挂载机械硬盘,开机就能自动启动/挂载 参考链接 https://blog.csdn.net/qq_35624642/article/details/137713143…...
Python轻量级 NoSQL 数据库之tinydb使用详解
概要 在现代应用开发中,使用数据库来存储和管理数据是非常常见的需求。对于简单的数据存储需求,关系型数据库可能显得过于复杂。TinyDB 是一个纯 Python 实现的轻量级 NoSQL 数据库,专为嵌入式场景设计,适用于小型项目、原型开发和教学等场景。本文将详细介绍 TinyDB 库,…...
【数据结构】二叉树(二)遍历
上篇已经了解对二叉树有了大概了解,本篇学习二叉树的前序、中序、后序及层序遍历的递归与非递归共7种遍历方法,快收藏吧~ 目录 1、前序遍历 递归方式: 迭代方式: 2、中序遍历 递归方式: 迭代方式: …...
NGINX 常用内置变量
目录 $remote_addr 变量 $args 变量 $is_args 变量 $document_root 变量 $document_uri 变量 $host 变量 $limit_rate 变量 $remote_port 变量 $remote_port --显示客户端端口 $request_method 变量 --返回请求方式 $request_filename 变量 --返回请求实际路径 $request_uri…...
Windows采用VS2019实现Open3D的C++应用
1、参考链接 https://blog.csdn.net/qq_31254435/article/details/137799739 但是,我的方法和上述链接不大一样,我是采用VS2019进行编译的,方便在Windows平台上验证各种算法。 2、创建一个VS2019的C Console工程 #include <iostream>…...
冒泡排序、选择排序、插入排序,三种简单排序算法的区别?
1、冒泡排序 冒泡排序是从下标 1 遍历到 n,每当遇到大于下一个的,就和上一个交换位置,这样最大的就移动到了 n 的位置,然后从头再从 1 遍历到 n-1,把第二大的移动到 n-1 的位置,依此类推,每次从…...
Docker 日志管理
一、ELK -Filebeat Elasticsearch 数据的存储和检索 常用端口: 9100:elasticsearch-head提供web访问 9200:elasticsearch与其他程序连接或发送消息 9300:elasticsearch集群状态 Logstash 有三个组件构成input,fi…...
JavaScript初级——基础知识
一、JS的HelloWord 1、JS的代码需要编写到script标签中 2、JS的执行是根据语句从上到下一次执行的。 二、JS的编写位置 1、可以将js代码编写到标签的onclick属性中,当我们点击按钮时,js代码才会执行。 2、可以将js代码写在超链接的href属性中࿰…...
0817(持久层框架:JDBC,MyBatis)
三层架构(表现层,业务层,持久层) java中框架的概述(表现层、业务层、持久层的关系)_控制层业务层持久层的关系-CSDN博客 框架:框架一般处在低层应用平台(如J2EE)和高层…...
在亚马逊云科技上安全、合规地创建AI大模型训练基础设施并开发AI应用服务
项目简介: 小李哥将继续每天介绍一个基于亚马逊云科技AWS云计算平台的全球前沿AI技术解决方案,帮助大家快速了解国际上最热门的云计算平台亚马逊云科技AWS AI最佳实践,并应用到自己的日常工作里。 本次介绍的是如何在亚马逊云科技利用Servi…...
无人机模拟训练室技术详解
无人机模拟训练室作为现代无人机技术培训的重要组成部分,集成了高精度模拟技术、先进的数据处理能力及高度交互的操作界面,为无人机操作员提供了一个安全、高效、接近实战的训练环境。以下是对无人机模拟训练室技术的详细解析,涵盖系统基础概…...
【Spring框架】
一、引言二、Spring核心概念三、Spring入门示例四、进一步了解Spring的依赖注入五、Spring的面向切面编程(AOP)六、总结 一、引言 Spring框架自2003年发布以来,凭借其轻量级、易于扩展的特性,在Java企业级应用开发领域得到了广泛…...
uniapp 日常业务 随便写写 源码
现成的组件 直接用 <template><view style"margin: 10rpx;"><view class"tea-header"><text class"tea-title">礼尚往来</text><view class"tea-view-all"><text>查看全部</text>&l…...
【软件测试】单元测试20套练习题
(一)概述 使用Java语言编写应用程序,设计测试数据,完成指定要求的白盒测试,对测试数据及相应测试结果进行界面截图,将代码以及相关截图粘贴到白盒测试报告中。 (二)题目要求...
RestClient
什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端,它允许HTTP与Elasticsearch 集群通信,而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级ÿ…...
PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建
制造业采购供应链管理是企业运营的核心环节,供应链协同管理在供应链上下游企业之间建立紧密的合作关系,通过信息共享、资源整合、业务协同等方式,实现供应链的全面管理和优化,提高供应链的效率和透明度,降低供应链的成…...
解决Ubuntu22.04 VMware失败的问题 ubuntu入门之二十八
现象1 打开VMware失败 Ubuntu升级之后打开VMware上报需要安装vmmon和vmnet,点击确认后如下提示 最终上报fail 解决方法 内核升级导致,需要在新内核下重新下载编译安装 查看版本 $ vmware -v VMware Workstation 17.5.1 build-23298084$ lsb_release…...
蓝桥杯 2024 15届国赛 A组 儿童节快乐
P10576 [蓝桥杯 2024 国 A] 儿童节快乐 题目描述 五彩斑斓的气球在蓝天下悠然飘荡,轻快的音乐在耳边持续回荡,小朋友们手牵着手一同畅快欢笑。在这样一片安乐祥和的氛围下,六一来了。 今天是六一儿童节,小蓝老师为了让大家在节…...
连锁超市冷库节能解决方案:如何实现超市降本增效
在连锁超市冷库运营中,高能耗、设备损耗快、人工管理低效等问题长期困扰企业。御控冷库节能解决方案通过智能控制化霜、按需化霜、实时监控、故障诊断、自动预警、远程控制开关六大核心技术,实现年省电费15%-60%,且不改动原有装备、安装快捷、…...
江苏艾立泰跨国资源接力:废料变黄金的绿色供应链革命
在华东塑料包装行业面临限塑令深度调整的背景下,江苏艾立泰以一场跨国资源接力的创新实践,重新定义了绿色供应链的边界。 跨国回收网络:废料变黄金的全球棋局 艾立泰在欧洲、东南亚建立再生塑料回收点,将海外废弃包装箱通过标准…...
ffmpeg(四):滤镜命令
FFmpeg 的滤镜命令是用于音视频处理中的强大工具,可以完成剪裁、缩放、加水印、调色、合成、旋转、模糊、叠加字幕等复杂的操作。其核心语法格式一般如下: ffmpeg -i input.mp4 -vf "滤镜参数" output.mp4或者带音频滤镜: ffmpeg…...
【算法训练营Day07】字符串part1
文章目录 反转字符串反转字符串II替换数字 反转字符串 题目链接:344. 反转字符串 双指针法,两个指针的元素直接调转即可 class Solution {public void reverseString(char[] s) {int head 0;int end s.length - 1;while(head < end) {char temp …...
【2025年】解决Burpsuite抓不到https包的问题
环境:windows11 burpsuite:2025.5 在抓取https网站时,burpsuite抓取不到https数据包,只显示: 解决该问题只需如下三个步骤: 1、浏览器中访问 http://burp 2、下载 CA certificate 证书 3、在设置--隐私与安全--…...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果