每日算法刷题Day31 6.14:leetcode二分答案2道题,结束二分答案,开始枚举技巧,用时1h10min

7. 1439.有序矩阵中的第K个最小数组和(困难,学习转化为373)

1439. 有序矩阵中的第 k 个最小数组和 - 力扣(LeetCode)

思想

1.给你一个 m * n 的矩阵 mat,以及一个整数 k ,矩阵中的每一行都以非递减的顺序排列。

你可以从每一行中选出 1 个元素形成一个数组。返回所有可能数组中的第 k 个 最小 数组和。

2.转化为373.查找和最小的K对数字,利用最小堆,373是从两个数组找前K个,而此题是m*n矩阵,但是发现假设已经取完矩阵前两行的数组和,再考虑第3行时,只要考虑前两行数组前K个值即可(因为后面的不可能是最终的K个最小数组和),所以问题就转化为得到前面i-1行的最小K个数组和数组,然后第i行考虑进来,最终再得到一个最小K个数组和数组,实现行的压缩

3.初始数组为只有0元素的数组和第一行(表示取第一行前K个元素)

代码

c++:

复制代码
class Solution {
public:
    vector<int> kSmallestPairs(vector<int>& nums1, vector<int>& nums2, int k) {
        int n1 = nums1.size(), n2 = nums2.size();
        priority_queue<tuple<int, int, int>> pq;
        vector<int> res;
        for (int i = 0; i < min(n1, k); ++i) {
            pq.emplace(-nums1[i] - nums2[0], i,
                       0); 
        }
        while (!pq.empty() && res.size() < k) {
            auto t = pq.top();
            pq.pop();
            int i = get<1>(t), j = get<2>(t);
            res.push_back(nums1[i] + nums2[j]);
            if (j + 1 < n2)
                pq.emplace(-nums1[i] - nums2[j + 1], i,
                           j + 1); 
        }
        return res;
    }
    int kthSmallest(vector<vector<int>>& mat, int k) {
        int n = mat.size();
        vector<int> ini = {0};
        for (auto& row : mat) {
            ini = kSmallestPairs(row, ini, k);
        }
        return ini.back();
    }
};
8. 786. 第K个最小的质数分数(中等)
思想

1.给你一个按递增顺序排序的数组 arr 和一个整数 k 。数组 arr1 和若干 质数 组成,且其中所有整数互不相同。

对于每对满足 0 <= i < j < arr.lengthij ,可以得到分数 arr[i] / arr[j]

那么第 k 个最小的分数是多少呢? 以长度为 2 的整数数组返回你的答案, 这里 answer[0] == arr[i]answer[1] == arr[j]

2.依旧转化为373.查找和最小的K对数字,只不过nums2是倒序的arr,且多个条件i+j!=n-1

代码

c++:

复制代码
class Solution {
public:
    vector<int> kthSmallestPrimeFraction(vector<int>& arr, int k) {
        int n = arr.size();
        vector<int> arr2 = arr;
        vector<vector<int>> res;
        reverse(arr2.begin(), arr2.end());
        priority_queue<tuple<double, int, int>> pq;
        for (int i = 0; i < min(n - 1, k); ++i)
            pq.emplace(-1.0 * arr[i] / arr2[0], i, 0);
        while (res.size() < k && !pq.empty()) {
            auto t = pq.top();
            pq.pop();
            int i = get<1>(t), j = get<2>(t);
            if (i + j == n - 1)
                continue;
            res.push_back({arr[i], arr2[j]});
            if (j + 1 < n)
                pq.emplace(-1.0 * arr[i] / arr2[j + 1], i, j + 1);
        }
        return res.back();
    }
};
相关推荐
MongoVIP5 分钟前
AI提示词应用
人工智能·职场和发展·简历优化·简历制作
顾你&11 分钟前
机器学习之无监督学习算法大总结
学习·算法·机器学习
神龙斗士2401 小时前
Java 数组的定义与使用
java·开发语言·数据结构·算法
Y.O.U..1 小时前
力扣HOT100-跳跃游戏II
算法·leetcode
hn小菜鸡1 小时前
LeetCode 3132.找出与数组相加的整数 II
算法·leetcode·职场和发展
微笑尅乐1 小时前
数组模拟加法——力扣66.加一
算法·leetcode·职场和发展
_不会dp不改名_1 小时前
leetcode_146 LRU缓存
算法·leetcode·缓存
aloha_7892 小时前
新国都面试真题
jvm·spring boot·spring·面试·职场和发展
帅帅爱数学2 小时前
DeepMimic论文详细解析:基于示例引导的深度强化学习实现物理仿真角色技能
算法·强化学习
Dream it possible!2 小时前
LeetCode 面试经典 150_哈希表_快乐数(45_202_C++_简单)(哈希表;快慢指针)
leetcode·面试·散列表