LeetCode-M373-堆-查找和最小的K对数字

题目

给定两个以升序排列的整形数组 nums1 和 nums2, 以及一个整数 k。定义一对值 (u,v),其中第一个元素来自 nums1,第二个元素来自 nums2。找到和最小的 k 对数字 (u1,v1), (u2,v2) ... (uk,vk)。
示例1:

输入: nums1 = [1,7,11], nums2 = [2,4,6], k = 3
输出: [1,2],[1,4],[1,6]
解释: 返回序列中的前 3 对数:
[1,2],[1,4],[1,6],[7,2],[7,4],[11,2],[7,6],[11,4],[11,6]

示例2:

输入: nums1 = [1,1,2], nums2 = [1,2,3], k = 2
输出: [1,1],[1,1]
解释: 返回序列中的前 2 对数:
[1,1],[1,1],[1,2],[2,1],[1,2],[2,2],[1,3],[1,3],[2,3]

示例3:

输入: nums1 = [1,2], nums2 = [3], k = 3
输出: [1,3],[2,3]
解释: 也可能序列中所有的数对都被返回:
[1,3],[2,3]

思路

遇到求第k大,第k小,前k大,前k小,就要想到优先对列,即最大堆或最小堆。求前k小,就可以通过维护一个k容量的最大堆,遍历完后堆中就是当前数据流中最小的k个元素;反之,求前k小,就维护一个k容量的最小堆,遍历完后堆中就是当前数据量中最大的k个元素。而对于第k大(第k小),也可以通过维护一个k容量的最小堆最大堆),遍历完后,堆顶即为第k大(小)元素。

思路1:这一题求前k小,即可以用最大堆来完成。但对于此题该解法存在一个效率问题,我们相当于暴力的将两个数组组合的所有情况压入堆中,会造成时间复杂度大大提高。
思路2:重新审查题目,可以看到题目给出了一个有效的前提,即每个数组递增排列。所以针对此情况,我们换个角度使用优先队列。
如果我们当前弹出的值是nums1[i],nums2[j], 那么我们下一个要添加的值应该是 num1[i],num2[j+1]。

如nums1 = {1,7,11,16} nums2 = {2,9,10,15},遍历nums1,分别与nums2的每一位元素组合,可以表示为:
(1,2) -> (1,9) -> (1,10) -> (1,15)
(7,2) -> (7,9) -> (7,10) -> (7,15)
(11,2) -> (11,9) -> (11,10) -> (11,15)
(16,2) -> (16,9) -> (16,10) -> (16,15)
针对上面这种表示方法,可以首先把一个列入堆。弹出一个后,下一个加入堆的是当前链表的下一个数字。

算法如下

  • 维护一个最小堆;
  • 先将第一列压入堆;
  • 取出堆顶第一个元素nums1[i],nums2[j],然后将nums1[i],nums2[j+1]压入堆;
  • 循环第二步,直到取出k个值或者当前堆为空。

解答

暴力解法

class Solution {
public:
    typedef vector<int> vec;
    struct cmp{  //自定义最大堆比较方式
        bool operator()(const vec &left, const vec &right) const {
            return (left[0]+left[1]) < (right[0]+right[1]);
        }
    };
public:
    vector<vec> kSmallestPairs(vec& nums1, vec& nums2, int k) {
        if(nums1.empty() || nums2.empty()) return {};
        //存在k大于可组合的最大数量
        k = k < nums1.size()*nums2.size() ? k : nums1.size()*nums2.size();
        vec temp(2,0);
        vector<vec> ans(k,temp);
        priority_queue<vec, vector<vec>, cmp> que;
        for(auto i : nums1) {  //暴力将所有情况压入堆
            for(auto j : nums2){
                temp[0] = i;
                temp[1] = j;
                que.push(temp);   
                if(que.size() > k) que.pop();
            }
        }
        while(k > 0){  //倒序输出
            ans[k-1] = que.top();
            que.pop();
            k--;
        }
        return ans;
    }
};

索引法

class Solution {
public:
    typedef vector<int> vec;
    struct data {  
        int num1;  //nums1的索引
        int num2;  //nums2的索引
        int sum;   //保存两个数组中元素的和
        data(int n1, int n2, int s): num1(n1), num2(n2), sum(s) {}
    };
    struct cmp { //自定义最小堆比较方式
        bool operator()(const data& left, const data& right) const { 
            return left.sum > right.sum;
        }
    };
public:
    vector<vec> kSmallestPairs(vec& nums1, vec& nums2, int k) {
        int len1 = nums1.size();
        int len2 = nums2.size();
        if(len1 < 1 || len2 < 1) return {};

        vector<vec> ans;
        priority_queue<data, vector<data>, cmp> que; //自定义最小堆
        for(int i=0; i<len1; ++i){
            //将nums1数组压入堆中
            que.push(data(i, 0, nums1[i] + nums2[0]));
        }
        while(!que.empty() && k > 0) {
            data top = que.top();  //保存堆顶
            //首先将nums1与nums2中第一个元素输出
            ans.push_back({nums1[top.num1], nums2[top.num2]});
            que.pop();
            //若当前堆顶中第二个数组的索引未到最后,
            //就将后面一个元素重新压入堆中
            if(top.num2 < len2-1) 
                que.push(data(top.num1, top.num2+1, 
                              nums1[top.num1] + nums2[top.num2+1]));
            k--;
        }
        
        return ans;
    }
};

【关注公众号DoCode,每日一道LeetCode,将零碎时间利用起来】

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容