当前位置: 首页 > news >正文

Java实现K个排序链表的高效合并:逐一合并、分治法与优先队列详解

Java实现K个排序链表的高效合并:逐一合并、分治法与优先队列详解

在算法和数据结构的学习中,链表是一个非常基础但又极具挑战的数据结构。尤其是当面对合并多个排序链表的问题时,如何在保证效率的前提下实现代码的简洁与高效,往往是算法设计的关键。本文将详细探讨如何使用Java中的优先队列(PriorityQueue)来实现K个排序链表的高效合并,并为常用的解决方案提供示例代码。

1. 问题背景

在这个问题中,我们有K个已经排序的链表,目标是将这些链表合并成一个新的排序链表,并返回合并后的链表头节点。简单的例子如下:

  • 链表1:1 -> 4 -> 5
  • 链表2:1 -> 3 -> 4
  • 链表3:2 -> 6

合并后的链表为:1 -> 1 -> 2 -> 3 -> 4 -> 4 -> 5 -> 6

这个问题的难点在于如何高效地处理K个链表的合并。以下是三种常用的解决方案。

2. 常用解决方案
  1. 逐一合并法

逐一合并法的思路是依次合并两个链表,最终得到结果链表。虽然这种方法简单易懂,但其时间复杂度较高,尤其是在链表数量较多的情况下。

示例代码:

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 MergeKListsSequentially {public ListNode mergeTwoLists(ListNode l1, ListNode l2) {ListNode dummy = new ListNode(0);ListNode tail = dummy;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.next = (l1 != null) ? l1 : l2;return dummy.next;}public ListNode mergeKLists(ListNode[] lists) {if (lists == null || lists.length == 0) {return null;}ListNode mergedList = lists[0];for (int i = 1; i < lists.length; i++) {mergedList = mergeTwoLists(mergedList, lists[i]);}return mergedList;}public static void main(String[] args) {ListNode list1 = new ListNode(1, new ListNode(4, new ListNode(5)));ListNode list2 = new ListNode(1, new ListNode(3, new ListNode(4)));ListNode list3 = new ListNode(2, new ListNode(6));ListNode[] lists = new ListNode[]{list1, list2, list3};MergeKListsSequentially solution = new MergeKListsSequentially();ListNode mergedList = solution.mergeKLists(lists);while (mergedList != null) {System.out.print(mergedList.val + " ");mergedList = mergedList.next;}}
}

解释

  • 通过mergeTwoLists方法,将两个有序链表合并成一个新的有序链表。
  • mergeKLists方法中,通过循环依次合并所有链表。

时间复杂度
最坏情况下为O(KN),其中K是链表的数量,N是每个链表的平均长度。

  1. 分治法

分治法是一种更高效的解决方案,通过递归地将链表分成两部分,分别进行合并,最终得到完整的合并链表。

示例代码:

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 MergeKListsDivideAndConquer {public ListNode mergeTwoLists(ListNode l1, ListNode l2) {ListNode dummy = new ListNode(0);ListNode tail = dummy;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.next = (l1 != null) ? l1 : l2;return dummy.next;}public ListNode mergeKLists(ListNode[] lists, int start, int end) {if (start == end) {return lists[start];}int mid = start + (end - start) / 2;ListNode left = mergeKLists(lists, start, mid);ListNode right = mergeKLists(lists, mid + 1, end);return mergeTwoLists(left, right);}public ListNode mergeKLists(ListNode[] lists) {if (lists == null || lists.length == 0) {return null;}return mergeKLists(lists, 0, lists.length - 1);}public static void main(String[] args) {ListNode list1 = new ListNode(1, new ListNode(4, new ListNode(5)));ListNode list2 = new ListNode(1, new ListNode(3, new ListNode(4)));ListNode list3 = new ListNode(2, new ListNode(6));ListNode[] lists = new ListNode[]{list1, list2, list3};MergeKListsDivideAndConquer solution = new MergeKListsDivideAndConquer();ListNode mergedList = solution.mergeKLists(lists);while (mergedList != null) {System.out.print(mergedList.val + " ");mergedList = mergedList.next;}}
}

解释

  • mergeKLists方法通过分治的方式递归地将链表两两合并,最终得到合并的结果链表。

时间复杂度
时间复杂度为O(N log K),其中N是所有节点的总数,K是链表的数量。

  1. 优先队列法

使用优先队列可以更加高效地解决问题。优先队列可以自动维护一个有序的队列,确保每次取出最小的节点,进行合并。

示例代码

import java.util.PriorityQueue;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 MergeKListsUsingPriorityQueue {public ListNode mergeKLists(ListNode[] lists) {PriorityQueue<ListNode> queue = new PriorityQueue<>((o1, o2) -> o1.val - o2.val);// 将每个链表的头节点加入优先队列for (ListNode list : lists) {if (list != null) {queue.add(list);}}// 创建虚拟头节点ListNode dummy = new ListNode(0);ListNode tail = dummy;// 逐步处理优先队列中的节点while (!queue.isEmpty()) {ListNode minNode = queue.poll();tail.next = minNode;tail = tail.next;// 如果最小节点的下一个节点不为空,继续加入优先队列if (minNode.next != null) {queue.add(minNode.next);}}// 防止链表成环,尾节点指向nulltail.next = null;return dummy.next;}public static void main(String[] args) {ListNode list1 = new ListNode(1, new ListNode(4, new ListNode(5)));ListNode list2 = new ListNode(1, new ListNode(3, new ListNode(4)));ListNode list3 = new ListNode(2, new ListNode(6));ListNode[] lists = new ListNode[]{list1, list2, list3};MergeKListsUsingPriorityQueue solution = new MergeKListsUsingPriorityQueue();ListNode mergedList = solution.mergeKLists(lists);while (mergedList != null) {System.out.print(mergedList.val + " ");mergedList = mergedList.next;}}
}

解释

  • mergeKLists方法使用优先队列存储每个链表的头节点,并逐步处理优先队列中的最小节点,确保合并后的链表依然有序。

时间复杂度
时间复杂度为O(N log K),其中N是所有节点的总数,K是链表的数量。
4. 优化
将每个链表的头节点插入优先队列,并通过逐步处理队列中的最小节点来维护链表的顺序。每次从队列中取出一个节点后,我们将该节点的下一个节点(如果存在)加入队列。这样可以减少优先队列的操作次数,从而提高效率。
示例代码

import java.util.PriorityQueue;class MergeKSortedListsOptimized {public ListNode mergeKLists1(ListNode[] lists) {// 创建一个最小堆优先队列,根据节点的值进行排序PriorityQueue<ListNode> queue = new PriorityQueue<>((o1, o2) -> o1.val - o2.val);// 遍历所有链表for (ListNode list : lists) {// 如果链表不为空if (list != null) {// 将链表头节点加入优先队列queue.add(list);}}// 创建一个虚拟头节点ListNode dummy = new ListNode(0);// 尾节点指向虚拟头节点ListNode tail = dummy;// 当优先队列不为空时while (!queue.isEmpty()) {// 从优先队列中取出最小的节点ListNode minNode = queue.poll();// 将最小节点连接到当前链表的末尾tail.next = minNode;// 更新尾节点为当前最小节点tail = tail.next;// 如果最小节点还有下一个节点if (minNode.next != null) {// 将最小节点的下一个节点加入优先队列queue.add(minNode.next);}}// 将尾节点的next指针置为null,表示链表结束tail.next = null;// 返回合并后的链表的头节点(虚拟头节点的下一个节点)return dummy.next;}
}

解释
该实现与前一个方法类似,但不同的是我们只将链表的头节点加入优先队列,然后逐个处理节点。在处理每个节点时,如果它有下一个节点,我们将下一个节点加入队列。这样做可以有效减少优先队列的操作次数,提高算法效率。

时间复杂度
时间复杂度为 O(N * log(K)),其中 K 是链表的数量,N 是所有链表中节点的总数。

4. 总结

通过以上三种常用的解决方案,我们可以清楚地看到,不同的方法在时间复杂度和实现复杂度上的权衡。对于链表数量较少的情况,逐一合并法可能更简单易行;而在链表数量较多时,分治法和优先队列法则可以更高效地解决问题。在实际开发中,优先队列法常常是首选,因为它在效率和实现上都表现得十分优秀。

相关文章:

Java实现K个排序链表的高效合并:逐一合并、分治法与优先队列详解

Java实现K个排序链表的高效合并&#xff1a;逐一合并、分治法与优先队列详解 在算法和数据结构的学习中&#xff0c;链表是一个非常基础但又极具挑战的数据结构。尤其是当面对合并多个排序链表的问题时&#xff0c;如何在保证效率的前提下实现代码的简洁与高效&#xff0c;往往…...

Xinstall揭秘:高效App推广背后的黑科技

在移动互联网时代&#xff0c;App的运营推广成为了开发者们最为关注的话题之一。然而&#xff0c;随着市场竞争的加剧&#xff0c;推广难度也越来越大。这时候&#xff0c;一款名为Xinstall的品牌走进了我们的视线&#xff0c;它以其独特的技术和解决方案&#xff0c;为App推广…...

星巴克VS瑞幸,新王、旧王之争给新CEO带来哪些启示

“变化是生命的元素&#xff0c;求变是生命的力量”&#xff0c;马克吐温曾这样解释生命。而商场上何尝不是如此&#xff0c;正因为世异则事异&#xff0c;求变也是企业的常态。 近日&#xff0c;星巴克官宣布莱恩尼科尔(Brian Niccol)将于9月9日接替拉什曼纳拉辛汉(Laxman Na…...

C语言 | Leetcode C语言题解之第354题俄罗斯套娃信封问题

题目&#xff1a; 题解&#xff1a; int cmp(int** a, int** b) {return (*a)[0] (*b)[0] ? (*b)[1] - (*a)[1] : (*a)[0] - (*b)[0]; }int maxEnvelopes(int** envelopes, int envelopesSize, int* envelopesColSize) {if (envelopesSize 0) {return 0;}qsort(envelopes, …...

大型俄罗斯国际展览会介绍

今天分享一些在俄罗斯比较知名的国际展览会&#xff0c;以下展会大多为一年一届&#xff0c;具体展出时间大家可去网上查询了解&#xff0c;充分利用合适的展会&#xff0c;无疑是企业拓展市场、塑造品牌、对接资源、紧跟行业趋势的重要途径。 1、俄罗斯国际食品展览会 展览地…...

CST软件仿真案例:圆极化平板天线仿真02

本期继续完成一款圆极化Patch天线的仿真实例。读者可以完整的了解到怎么用CST微波工作室&#xff0c;完成对一款天线建模、设置到仿真分析的完整过程。 本期中&#xff0c;我们要设计的圆极化天线尺寸如下图所示&#xff1a; 本期内容是接着上期部分开始。首先先完成仿真实例0…...

【前端】vue监视属性和计算属性对比

首先分开讲解各个属性的作用。 1.计算属性 作用&#xff1a;用来计算出来一个值&#xff0c;这个值调用的时候不需要加括号&#xff0c;会根据依赖进行缓存&#xff0c;依赖不变&#xff0c;computed的值不会重新计算。 const vm new Vue({el:#root,data:{lastName:张,firstNa…...

探索提示工程 Prompt Engineering的奥妙

一、探索提示工程 Prompt Engineering 1. 介绍通用人工智能和专用人工智能 人工智能&#xff08;AI&#xff09;可以分为通用人工智能&#xff08;AGI&#xff09;和专用人工智能&#xff08;Narrow AI&#xff09;。AGI是一种能够理解、学习和执行任何人类可以完成的任务的智…...

算法阶段总结1

阶段总结 通过今天晚上的这场div2我深刻的意识到&#xff0c;光是会找窍门是远远不够的&#xff0c;你得会基础的建图&#xff0c;dp&#xff0c;高级数据结构&#xff0c;你这样才可以不断的提升自己&#xff0c;不可以一直在一个阶段停留下去&#xff0c;构造题可以刷下去&a…...

前端宝典之七:React性能优化实战精华篇

本文主要讲解实战项目中React性能优化的方法&#xff0c;主要分为三个大的方面&#xff1a;减少不必要的组件更新、组件优化以及tree-shaking&#xff0c;共11个方法 一、减少不必要组件更新 以下是一些可以避免在 React 提交阶段进行不必要重新渲染的方法&#xff1a; 1、使…...

【Dash】feffery_antd_components 简单入门示例

一、简单了解 feffery_antd_components 简称 fac &#xff0c;是一个基于 Ant Design 的 Dash 第三方组件&#xff0c;由Feffery 老师开源维护的 Python 网页开发组件库&#xff0c;它具有丰富的页面常用交互组件功能&#xff0c;使开发者可以使用纯Python的方式快速构建现代…...

JAVA学习-练习试用Java实现“路径交叉”

问题&#xff1a; 给定一个整数数组 distance 。从 X-Y 平面上的点 (0,0) 开始&#xff0c;先向北移动 distance[0] 米&#xff0c;然后向西移动 distance[1] 米&#xff0c;向南移动 distance[2] 米&#xff0c;向东移动 distance[3] 米&#xff0c;持续移动。也就是说&#…...

element组件封装

1.上传组件 <!--文件上传组件--> <template><div class"upload-file"><el-uploadref"fileUpload"v-if"props.type default":action"baseURL other.adaptationUrl(props.uploadFileUrl)":before-upload"h…...

Mysql (面试篇)

目录 唯一索引比普通索引快吗 MySQL由哪些部分组成&#xff0c;分别用来做什么 MySQL查询缓存有什么弊端&#xff0c;应该什么情况下使用&#xff0c;8.0版本对查询缓存由上面变更 MyISAM和InnoDB的区别有哪些 MySQL怎么恢复半个月前的数据 MySQL事务的隔离级别&#xff…...

【python】深入探讨python中的抽象类,创建、实现方法以及应用实战

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…...

微前端传值

在微前端架构中&#xff0c;不同子应用之间通过 postMessage 进行通信是一种常见的做法。这种方式允许不同源的窗口之间进行安全的信息交换。 下面是如何使用 postMessage 在微前端环境中发送和接收消息的示例。 步骤 1: 发送消息 假设您有一个主应用&#xff08;host app&a…...

《学会 SpringBoot · 依赖管理机制》

&#x1f4e2; 大家好&#xff0c;我是 【战神刘玉栋】&#xff0c;有10多年的研发经验&#xff0c;致力于前后端技术栈的知识沉淀和传播。 &#x1f497; &#x1f33b; CSDN入驻不久&#xff0c;希望大家多多支持&#xff0c;后续会继续提升文章质量&#xff0c;绝不滥竽充数…...

全网行为管理软件有哪些?5款总有一款适合你的企业!

如今企业越来越依赖互联网进行日常运营和业务发展&#xff0c;网络行为管理变得日益重要。 为了确保网络安全、提高员工工作效率、避免敏感信息外泄等问题&#xff0c;企业往往需要借助全网行为管理软件来监控和管理内部网络的使用情况。 本文将为您介绍五款热门的全网行为管理…...

以简单的例子从头开始建spring boot web多模块项目(二)-mybatis简单集成

继续使用以简单的例子从头开始建spring boot web多模块项目&#xff08;一&#xff09;中的项目进行mybatis集成。 1、pom.xml文件中&#xff0c;增加相关的依赖包的引入&#xff0c;分别是mybatis-spring-boot-starter、lombok、mysql-connector-java 如下&#xff1a; <d…...

Golang | Leetcode Golang题解之第354题俄罗斯套娃信封问题

题目&#xff1a; 题解&#xff1a; func maxEnvelopes(envelopes [][]int) int {n : len(envelopes)if n 0 {return 0}sort.Slice(envelopes, func(i, j int) bool {a, b : envelopes[i], envelopes[j]return a[0] < b[0] || a[0] b[0] && a[1] > b[1]})f : …...

Echarts 数据大屏实战:150套模板助力企业级可视化开发

1. 为什么企业需要Echarts数据大屏&#xff1f; 在数字化转型的浪潮中&#xff0c;数据可视化已经成为企业决策的重要工具。想象一下&#xff0c;当你的老板需要在3秒内了解公司当月销售情况、用户增长趋势和库存状态时&#xff0c;密密麻麻的Excel表格显然不是最佳选择。这时…...

零基础上手DownKyi:B站视频下载工具的高效使用指南

零基础上手DownKyi&#xff1a;B站视频下载工具的高效使用指南 【免费下载链接】downkyi 哔哩下载姬downkyi&#xff0c;哔哩哔哩网站视频下载工具&#xff0c;支持批量下载&#xff0c;支持8K、HDR、杜比视界&#xff0c;提供工具箱&#xff08;音视频提取、去水印等&#xff…...

生信分析效率翻倍:fastp多线程+UMI处理技巧全解析(含实战案例)

生信分析效率翻倍&#xff1a;fastp多线程UMI处理技巧全解析&#xff08;含实战案例&#xff09; 在肿瘤基因检测和ctDNA测序领域&#xff0c;数据预处理环节往往成为整个分析流程的瓶颈。传统工具在处理海量测序数据时&#xff0c;不仅耗时长达数小时&#xff0c;还经常面临内…...

HUNYUAN-MT 7B翻译终端Matlab科学计算集成:技术文档跨语言协作

HUNYUAN-MT 7B翻译终端Matlab科学计算集成&#xff1a;技术文档跨语言协作 如果你在科研或工程团队里工作&#xff0c;很可能遇到过这样的场景&#xff1a;团队里有来自不同国家的同事&#xff0c;大家用Matlab写的算法注释、实验报告、技术文档&#xff0c;语言五花八门。你想…...

MediaPipe人体骨骼检测:零配置Web应用,上传图片秒出骨架图

MediaPipe人体骨骼检测&#xff1a;零配置Web应用&#xff0c;上传图片秒出骨架图 1. 引言&#xff1a;一键式骨骼检测的便捷体验 想象一下这样的场景&#xff1a;健身教练需要快速分析学员的动作姿势&#xff0c;医生希望直观展示患者的骨骼姿态&#xff0c;或者动画师需要参…...

EEGLAB进阶实战:从原始EEG到ERP成分的精准提取与可视化分析

1. EEGLAB入门&#xff1a;理解ERP分析的核心流程 第一次接触EEGLAB时&#xff0c;我被它强大的功能和复杂的界面弄得晕头转向。经过多次实战&#xff0c;我发现理解ERP分析的完整流程是关键。就像做菜需要先备料再烹饪一样&#xff0c;EEG数据处理也需要遵循特定步骤。 原始EE…...

FireRedASR-AED-L语音搜索应用:电商场景实战

FireRedASR-AED-L语音搜索应用&#xff1a;电商场景实战 1. 引言 想象一下这个场景&#xff1a;一位正在做饭的用户手上沾满面粉&#xff0c;突然想起需要购买烘焙材料&#xff0c;只需对着手机说"帮我找高筋面粉"&#xff0c;下一秒就能看到精准的商品搜索结果。这…...

技术破局:B端拓客号码核验的痛点突围与行业新生态,氪迹科技法人股东 核验筛选系统,阶梯式价格

在B端拓客进入“精准致胜”的新时代&#xff0c;线索质量直接决定拓客成效&#xff0c;而号码核验作为筛选有效线索的“第一道门槛”&#xff0c;其服务水平直接影响拓客团队的投入回报与运营效率。当下&#xff0c;随着AI拓客技术的普及&#xff0c;号码核验已渗透到电销、金融…...

OpenClaw多环境部署:GLM-4.7-Flash开发与生产配置

OpenClaw多环境部署&#xff1a;GLM-4.7-Flash开发与生产配置 1. 为什么需要区分开发与生产环境 去年我在尝试用OpenClaw自动化处理公司内部文档时&#xff0c;踩过一个典型的坑&#xff1a;直接在开发机上配置的生产环境参数&#xff0c;导致测试脚本误删了正式服务器上的文…...

抖音弹幕抓取终极指南:如何利用系统代理技术实现免费数据监听

抖音弹幕抓取终极指南&#xff1a;如何利用系统代理技术实现免费数据监听 【免费下载链接】DouyinBarrageGrab 基于系统代理的抖音弹幕wss抓取程序&#xff0c;能够获取所有数据来源&#xff0c;包括chrome&#xff0c;抖音直播伴侣等&#xff0c;可进行进程过滤 项目地址: h…...