IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> java - 23. 合并K个升序链表 - 优先级队列PriorityQueue -> 正文阅读

[数据结构与算法]java - 23. 合并K个升序链表 - 优先级队列PriorityQueue

一 思路

1 和合并2个升序链表类似,先对比链表头的k个数,这里用优先级队列(二叉树堆)来保存。
2 每次拿出二叉树堆中的最小值(即k个数中的最小值),然后把当前链表的下一个节点入堆,始终保证是k个链表头,依次取最小值,直到排序完成

二 代码

class Solution {
    // 第一种写法:lambda表达式
    public ListNode mergeKLists(ListNode[] lists) {
        if (lists.length == 0) return null;
        PriorityQueue<ListNode> tempQueue = new PriorityQueue<>((o1, o2) -> o1.val - o2.val);  //用于存放k个待排序的链表头
        ListNode dummy = new ListNode(-1);  // 虚拟头节点
        ListNode p = dummy;
        for (ListNode head : lists){
            if (head != null){
                tempQueue.add(head); // 拿到k个链表头
            }
        }

        while (!tempQueue.isEmpty()){
            p.next = tempQueue.poll();
            if (p.next.next != null){
                tempQueue.add(p.next.next);
            }
            p = p.next;
        }
        return dummy.next;
    }


-----------------------------------------------------------------------------------------
   
    // 第二种写法: 匿名类
    public ListNode mergeKLists(ListNode[] lists) {
        if (lists.length == 0) return null;
        //用于存放k个待排序的链表头
        PriorityQueue<ListNode> tempQueue = new PriorityQueue<>(new Comparator<ListNode>() {
            @Override
            public int compare(ListNode o1, ListNode o2) {
                return o1.val - o2.val;
            }
        });
        ListNode dummy = new ListNode(-1);  // 虚拟头节点
        ListNode p = dummy;
        for (ListNode head : lists){
            if (head != null){
                tempQueue.add(head); // 拿到k个链表头
            }
        }

        while (!tempQueue.isEmpty()){
            ListNode node = tempQueue.poll();
            p.next = node;
            if (p.next.next != null){
                tempQueue.add(p.next.next);
            }
            p = p.next;
        }
        return dummy.next;
    }
}

三 说明

????????1 通过看源码可以发现,优先级队列类一共提供了多种构造器,无参构造是默认队列长度为全局常量DEFAULT_INITIAL_CAPACITY,值为11。

????????比较器默认为null。其他有参构造器均可用户指定,形参为优先级队列initialCapacity和比较器comparator。

    public PriorityQueue() {
        this(DEFAULT_INITIAL_CAPACITY, null);
    }

    public PriorityQueue(int initialCapacity) {
        this(initialCapacity, null);
    }

    public PriorityQueue(Comparator<? super E> comparator) {
        this(DEFAULT_INITIAL_CAPACITY, comparator);
    }
    
    public PriorityQueue(int initialCapacity,
                         Comparator<? super E> comparator) {
        // Note: This restriction of at least one is not actually needed,
        // but continues for 1.5 compatibility
        if (initialCapacity < 1)
            throw new IllegalArgumentException();
        this.queue = new Object[initialCapacity];
        this.comparator = comparator;
    }

????????2 因此本题在初始化优先级队列的时候可以指定容量,也可以自动扩容。

????????但是比较器默认是按照自然顺序natural ordering 来排序,这肯定不是这道题的需求,这道题需要按照链表节点的值的大小来排序,这时候需要我们自定义比较器Comparator。

????????Comparator是一个接口,其中定义了抽象的compare方法。本题通过重写该方法达到比较链表节点值的目的,因此便有了解法二的定义匿名类重写compare方法的写法:

PriorityQueue<ListNode> tempQueue = new PriorityQueue<>(new Comparator<ListNode>() {
            @Override
            public int compare(ListNode o1, ListNode o2) {
                return o1.val - o2.val;
            }
        });

还有解法一lambda表达式的写法,两者的效果都一样。

PriorityQueue<ListNode> tempQueue = new PriorityQueue<>((o1, o2) -> o1.val - o2.val);

根据比较器的规则,优先级队列将变为小顶堆或者大顶堆。

小顶堆,形参(o1, o2)??return?前减后(o1-o2);
大顶堆,?形参(o1, o2)?return?后减前(o2-o1);

????????

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-05-10 12:08:32  更:2022-05-10 12:11:27 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/1 22:28:27-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码