力扣面试150 查找和最小的 K 对数字 最小堆 去重

Problem: 373. 查找和最小的 K 对数字

👨‍🏫 参考题解

Java 复制代码
class Solution {
    public List<List<Integer>> kSmallestPairs(int[] nums1, int[] nums2, int k) {
        // 创建一个大小为 k 的结果列表,用于存储和最小的 k 个数对
        List<List<Integer>> ans = new ArrayList<>(k); // 预分配空间
        
        // 创建一个优先队列(小根堆),存储三元组 [nums1[i] + nums2[j], i, j]
        // 按照和 (nums1[i] + nums2[j]) 的大小升序排列
        PriorityQueue<int[]> pq = new PriorityQueue<>((a, b) -> a[0] - b[0]);
        
        // 将 nums1 中前 k 个元素与 nums2 中第一个元素的和及其索引 i, j 加入到优先队列中
        for (int i = 0; i < Math.min(nums1.length, k); i++) { // 至多 k 个
            pq.add(new int[]{nums1[i] + nums2[0], i, 0});
        }
        
        // 循环直到找到 k 个数对或者优先队列为空
        while (ans.size() < k && !pq.isEmpty()) {
            // 取出堆顶元素,也就是当前和最小的数对
            int[] p = pq.poll();
            int i = p[1]; // 取出 nums1 的索引
            int j = p[2]; // 取出 nums2 的索引
            
            // 将当前和最小的数对加入结果列表
            ans.add(List.of(nums1[i], nums2[j]));
            
            // 如果 nums2 中还有剩余元素,将新的数对 [nums1[i], nums2[j + 1]] 放入优先队列
            if (j + 1 < nums2.length) {
                pq.add(new int[]{nums1[i] + nums2[j + 1], i, j + 1});
            }
        }
        
        // 返回结果列表
        return ans;
    }
}
相关推荐
6Hzlia3 分钟前
【Hot 100 刷题计划】 LeetCode 152. 乘积最大子数组 | C++ 动态规划 (绝妙 swap 翻转技巧)
c++·leetcode·动态规划
smj2302_796826524 分钟前
解决leetcode第3901题好子序列查询
python·算法·leetcode
_深海凉_6 分钟前
LeetCode热题100-每日温度
算法·leetcode·职场和发展
迷你可可小生8 分钟前
面经学习(二)
学习·算法
John.Lewis9 分钟前
C++加餐课-二叉树:进阶算法
数据结构·c++·算法
郝学胜-神的一滴18 分钟前
ReLU激活函数全解析:从原理到实战,解锁深度学习核心激活单元
人工智能·pytorch·python·深度学习·算法
AGV算法笔记23 分钟前
最新感知算法论文分析:RaCFormer 如何提升雷达相机 3D 目标检测性能?
数码相机·算法·3d·自动驾驶·机器人视觉·3d目标检测·感知算法
李日灐23 分钟前
<3>Linux 基础指令:从时间、查找、文本过滤到 .zip/.tgz 压缩解压与常用热键
linux·运维·服务器·开发语言·后端·面试·指令
脱氧核糖核酸__24 分钟前
LeetCode热题100——54.螺旋矩阵(题解+答案+要点)
c++·算法·leetcode·矩阵
lxh011330 分钟前
电话号码的字母组合
java·javascript·算法