Java实现合并多个升序链表

 更新时间:2023年04月16日 10:32:41   作者:Java星辰  
本文主要介绍了Java实现合并多个升序链表,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

本文主要介绍如何将多个小的升序链表合并一个大的升序链表。

需求描述

给出K个升序链接,要求把这K个升序链表合并成一个,并且这个链表也是升序的。

例如:A = [1,5,6]B = [2,3,8], C = [4,4,9] 将这3个链表合并成一个链表D,合并后D = [1,2,3,4,4,5,6,8,9],并且将D的第一个节点返回。

思路解析

我们可以采用优先级队列来实现,先把每个链表的头结点放到一个优先级队列里,优先级队列也叫小根堆。

在放小根堆的时候,谁小就把谁放在最上面。需要注意的是,我们放入的时候,放入的是节点,所以通过这个节点是可以访问整个链表的。

我们看下处理过程:

  • 首先把每个链接的头结点放入小根堆中:1,2,4
  • 首先弹出最小的值:1
  • 1节点的下一个节点5放入小根堆中,此时小根堆会自动调整顺序,此时为:2, 4, 5
  • 2节点弹出,让1节点的next指针指向2节点,并且将2节点的下一个节点6放入小根堆,此时已弹出的节点为 1,2,而小根堆为4, 5, 6
  • 4节点弹出,让2节点的next指针指向4节点,并且将4节点的下一个节点4放入小根堆中,此时已弹出的节点为1,2,4,而小根堆为4, 5, 6
  • 依此类推,每弹出一个节点,拼接在已弹出节点的后面,并将弹出节点的下一个节点放入小根堆中,直到小根堆中所有的元素全部弹出。

好了,现在整体思路有了,但是现在是不是有个疑问?我们在做算法时,使用到了优先队列,那么我们可以使用系统自带的优先队列吗?

个人感觉,如果是面试时,这个系统自带的类只是题目中很小的一部分,比如上面的题目,主要考察的是如何实现这个过程,而不是考察如何实现优先队列的,如果没有特殊要求不让使用的话,是可以使用的。当然,如果考察是要实现一个优先队列,我要是直接new一个PriorityQueue,我估计面试官会一巴掌把我拍出来。

代码实现

链表节点定义如下:

public class ListNode {
   public int val;
   public ListNode next;
}

因为是小根堆,需要一个排序算法,所以定义一个比较器如下:

public class ListNodeComparator implements Comparator<ListNode> {
   @Override
   public int compare(ListNode o1, ListNode o2) {
      return o1.val - o2.val; 
   }
}

合并链接:

public ListNode mergeKLists(ListNode[] lists) {
   if (lists == null) {
      return null;
   }
   PriorityQueue<ListNode> heap = new PriorityQueue<>(new ListNodeComparator());
   for (int i = 0; i < lists.length; i++) {
      if (lists[i] != null) {
         heap.add(lists[i]);
      }
   }
   if (heap.isEmpty()) {
      return null;
   }
   ListNode head = heap.poll();
   ListNode pre = head;
   if (pre.next != null) {
      heap.add(pre.next);
   }
   while (!heap.isEmpty()) {
      ListNode cur = heap.poll();
      pre.next = cur;
      pre = cur;
      if (cur.next != null) {
         heap.add(cur.next);
      }
   }
   return head;
}

这个方法参数lists代表要传进来多少个链表,方法合并多个链表后,返回链表的第一个节点。

时间复杂度

假设有M个链表,M个链表的总节点个数为N。此时,对于小根堆来说,他的规模大小为M,则对于小根堆来说他的操作时间复杂度为O(logM),一共有N个节点,所以时间复杂度为O(N*logM)

总结

本文主要介绍如何将多个小的升序链表合并一个大的升序链表,介绍了实现这个功能的思路分析,使用优先队列自动排序的特性实现了这个功能,当然这里我们使用的是系统自带的优先队列,其实也可以自己实现一个,个人感觉没太必要,就先偷个懒 。更多相关Java 合并升序链表内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

到此这篇关于Java实现合并多个升序链表的文章就介绍到这了,更多相关Java 合并升序链表内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java通过卖票理解多线程

    Java通过卖票理解多线程

    本文主要介绍了一个多线程卖票的例子,通过卖票这个实例来介绍多线程的方式,加深理解,需要的朋友可以参考下
    2017-09-09
  • SpringBoot集成mqtt的多模块项目配置详解

    SpringBoot集成mqtt的多模块项目配置详解

    这篇文章主要介绍了SpringBoot集成mqtt的多模块项目配置详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • JavaEE多线程中阻塞队列的项目实践

    JavaEE多线程中阻塞队列的项目实践

    本文主要介绍了JavaEE多线程中阻塞队列的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-09-09
  • Java HashMap源码深入分析讲解

    Java HashMap源码深入分析讲解

    在java开发中,HashMap是最常用、最常见的集合容器类之一,下面一起温故一下,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-08-08
  • Java经典排序算法之插入排序

    Java经典排序算法之插入排序

    这篇文章主要为大家详细介绍了Java经典排序算法之插入排序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-04-04
  • java mail使用qq邮箱发邮件的配置方法

    java mail使用qq邮箱发邮件的配置方法

    本文为你介绍了java mail使用qq邮箱发邮件的方法,大家参考使用吧
    2014-01-01
  • 侦听消息队列的Message Listener类示例详解

    侦听消息队列的Message Listener类示例详解

    Spring AMQP 是基于 Spring 框架的AMQP消息解决方案,提供模板化的发送和接收消息的抽象层,提供基于消息驱动的 POJO的消息监听等,简化了我们对于RabbitMQ相关程序的开发,本文给大家介绍侦听消息队列的Message Listener类,感兴趣的朋友一起看看吧
    2023-12-12
  • Java事件机制要素及实例详解

    Java事件机制要素及实例详解

    这篇文章主要介绍了Java事件机制要素及实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • 定时任务注解@Scheduled不生效问题及解决

    定时任务注解@Scheduled不生效问题及解决

    这篇文章主要介绍了定时任务注解@Scheduled不生效问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-06-06
  • Spring @Value 设置默认值的实现

    Spring @Value 设置默认值的实现

    这篇文章主要介绍了Spring @Value 设置默认值的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09

最新评论