【经典算法题】查找和最小的K对数字

【经典算法题】查找和最小的K对数字

Leetcode 0373 查找和最小的K对数字

题目描述:Leetcode 0373 查找和最小的K对数字

在这里插入图片描述

分析

  • 本题的考点:多路归并、堆

  • 本题一共有 n × m n \times m n×m对数据,将数组nums1、nums2分别记为a、b,按照数组b可以将这些数据分为m路,且每一路都是升序的,如下图:

在这里插入图片描述

  • 使用小根堆求解前k小的数据即可。

代码

  • C++
typedef vector<int> VI;  // (a[i] + b[j], i, j)

class Solution {
public:
    vector<vector<int>> kSmallestPairs(vector<int> &a, vector<int> &b, int k) {

        if (a.empty() || b.empty()) return {};

        int n = a.size(), m = b.size();
        priority_queue<VI, vector<VI>, greater<VI>> heap;
        for (int i = 0; i < m; i++) heap.push({a[0] + b[i], 0, i});
        vector<VI> res;
        while (k-- && heap.size()) {
            auto t = heap.top(); heap.pop();
            res.push_back({a[t[1]], b[t[2]]});
            if (t[1] + 1 < n)
                heap.push({a[t[1] + 1] + b[t[2]], t[1] + 1, t[2]});
        }
        return res;
    }
};
  • Java
class Solution {
    public List<List<Integer>> kSmallestPairs(int[] a, int[] b, int k) {

        int n = a.length, m = b.length;
        if (n == 0 || m == 0) return new ArrayList<>();

        PriorityQueue<int[]> heap = new PriorityQueue<>((o1, o2) -> { return o1[0] - o2[0]; });
        for (int i = 0; i < m; i++) heap.add(new int[]{a[0] + b[i], 0, i});
        List<List<Integer>> res = new ArrayList<>();
        while (k-- != 0 && !heap.isEmpty()) {
            int[] t = heap.remove();
            res.add(Arrays.asList(a[t[1]], b[t[2]]));
            if (t[1] + 1 < n)
                heap.add(new int[]{a[t[1] + 1] + b[t[2]], t[1] + 1, t[2]});
        }
        return res;
    }
}

时空复杂度分析

  • 时间复杂度: O ( k × l o g ( n ) ) O(k \times log(n)) O(k×log(n))n为数组长度。

  • 空间复杂度: O ( n ) O(n) O(n)

扩展题目

AcWing 146. 序列

问题描述()

分析

  • 一共m个序列,每个序列都是n个数据,每个序列中选出一个数,求和。我们需要找到最小的n个。

  • 考虑前两个序列,两个序列可以构成 n 2 n^2 n2 个数据,我们找到前n个最小的数据即可,后面的数据一定用不到。因此对于这m个序列,每次合并两个序列得到一个新的序列,直到合并到只有一个序列就是最后的答案。

  • 考虑如何合并两个序列。对于数组a和数组b,我们可以让数组a升序排列,然后构造n路,使用n路归并排序即可。如下图:

在这里插入图片描述

代码

  • C++
#include <iostream>
#include <cstring>
#include <algorithm>
#include <queue>

#define x first
#define y second

using namespace std;

typedef pair<int, int> PII;

const int N = 2010;

int m, n;
int a[N], b[N];
int c[N];  // 多路归并过程中使用到的数组

void merge() {
    
    priority_queue<PII, vector<PII>, greater<PII>> heap;
    
    for (int i = 0; i < n; i++) heap.push({b[i] + a[0], 0});
    
    for (int i = 0; i < n; i++) {
        auto t = heap.top();
        heap.pop();
        c[i] = t.x;
        heap.push({t.x - a[t.y] + a[t.y + 1], t.y + 1});
    }
    memcpy(a, c, sizeof a);
}

int main() {
    
    int T;
    scanf("%d", &T);
    while (T--) {
        scanf("%d%d", &m, &n);
        for (int i = 0; i < n; i++) scanf("%d", &a[i]);
        
        sort(a, a + n);
        
        for (int i = 0; i < m - 1; i++) {
            for (int j = 0; j < n; j++) scanf("%d", &b[j]);
            merge();
        }
        
        for (int i = 0; i < n; i++) printf("%d ", a[i]);
        puts("");
    }
    
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
字节跳动常见算法面试top50整理如下: 1. 两数之和:给定一个整数数组和一个目标值,找出数组中和为目标值的两个数。 2. 两数相加:给定两个非空链表表示两个非负整数,将两数相加返回一个新的链表。 3. 无重复字符的最长子串:给定一个字符串,请找出其中不含有重复字符的最长子串的长度。 4. 两个排序数组的中位数:给定两个大小分别为 m 和 n 的有序数组 nums1 和 nums2,请找出这两个有序数组的中位数。 5. 电话号码的字母组合:给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。 6. 四数之和:给定一个包含 n 个整数的数组 nums 和一个目标值 target,判断 nums 中是否存在四个元素 a、b、c 和 d,使得 a + b + c + d 的值与 target 相等。 7. 合并两个有序链表:将两个有序链表合并为一个新的有序链表并返回。 8. 验证回文串:给定一个字符串,验证它是否是回文串。 9. 最长有效括号:给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度。 10. 二叉树的最大深度:给定一个二叉树,找出其最大深度。 11. 盛最多水的容器:给定 n 个非负整数 a1,a2,...,an,每个数代表坐标中的一个点 (i, ai)。在坐标内画 n 条垂直线,垂直线 i 的两个端点分别为 (i, ai) 和 (i, 0)。找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。 12. 三数之和:给定一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ? 13. 最接近的三数之和:给定一个包括 n 个整数的数组 nums 和 一个目标值 target。找出 nums 中的三个整数,使得它们的和与 target 最接近。 14. 有效的括号:给定一个只包括 '(',')','{','}','[',']' 的字符串,判断字符串是否有效。 15. 合并两个有序数组:给定两个有序数组 nums1 和 nums2,将 nums2 合并到 nums1 中,使得 num1 成为一个有序数组。 16. 数组中的第K个最大元素:在未排序的数组中找到第 k 个最大的元素。 17. 罗马数字转整数:将罗马数字转换成整数。 18. 最小路径和:给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和最小。 19. 矩阵置零:给定一个 m x n 的矩阵,如果一个元素为 0,则将其所在行和列的所有元素都设为 0。 20. 字符串相乘:给定两个以字符串形式表示的非负整数 num1 和 num2,返回 num1 和 num2 的乘积。 21.有效的数独:判断一个 9x9 的数独是否有效。 22. 旋转图像:给定一个 n × n 的二维矩阵表示一个图像,将图像顺时针旋转 90 度。 23. 搜索旋转排序数组:假设按照升序排序的数组在预先未知的某个点上进行了旋转。 24. 螺旋矩阵:给定一个包含 m x n 个元素的矩阵(m 行, n 列),请按照顺时针螺旋顺序,返回矩阵中的所有元素。 25. 合并K个排序链表:合并 k 个排序链表,返回合并后的排序链表。 26. 不同路径:一个机器人位于一个 m x n 网格的左上角,机器人每次只能向下或者向右移动一步。 27. 跳跃游戏:给定一个非负整数数组,你最初位于数组的第一个位置。 28. 插入区间:给出一个无重叠的,按照区间起始端点排序的区间列表。 29. 最长公共前缀:编写一个函数来查找字符串数组中的最长公共前缀。 30. 螺旋矩阵 II:给定一个正整数 n,生成一个包含 1 到 n2 所有元素,且元素按顺时针顺序螺旋排列的正方形矩阵。 31. 编辑距离:给定两个单词 word1 和 word2,计算出将 word1 转换成 word2 所使用的最少操作数。 32. 删除排序链表中的重复元素:给定一个排序链表,删除所有重复的元素,使得每个元素只出现一次。 33. 字符串转整数(atoi):请你来实现一个 atoi 函数,使其能将字符串转换成整数。 34. 平衡二叉树:给定一个二叉树,判断它是否是高度平衡的二叉树。 35. Pow(x, n):实现 pow(x, n),即计算 x 的 n 次幂函数。 36. 搜索二维矩阵:编写一个高效的算法来判断 m x n 矩阵中,是否存在一个目标值。 37. 接雨水:给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。 38. 二进制求和:给定两个二进制字符串,返回它们的和(用二进制表示)。 39. 括号生成:给出 n 对括号,请编写一个函数来生成所有的由 n 对括号组成的合法组合。 40. 逆波兰表达式求值:根据逆波兰表示法,求表达式的值。 41. 环形链表:给定一个链表,判断链表中是否有环。 42. 环形链表 II:给定一个链表,返回链表开始入环的第一个节点。 43. 重建二叉树:根据一棵树的前序遍历与中序遍历构造二叉树。 44. 验证二叉搜索树:给定一个二叉树,判断其是否是一个有效的二叉搜索树。 45. 二叉树的中序遍历:给定一个二叉树,返回它的中序 遍历。 46. 最小栈:设计一个支持 push,pop,top 操作,并能在常数时间内检索到最小元素的栈。 47. 单词拆分:给定一个非空字符串 s 和一个包含非空单词列表的字典 wordDict,判定 s 是否可以被空格拆分为一个或多个在字典中出现的单词。 48. 对称二叉树:给定一个二叉树,检查它是否是镜像对称的。 49. N皇后问:给定一个整数 n,返回所有不同的 n 皇后问的解决方案。 50. 跳跃游戏 II:给定一个非负整数数组,你最初位于数组的第一个位置,在该位 我们的主要任务是根据输入来模拟对话。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值