[M思维] lc264. 丑数 II(多路归并+STL堆)

1. 题目来源

链接:264. 丑数 II

已有题解:[剑指-Offer] 49. 丑数(思维、代码优化、巧妙解法)

2. 题目解析

一道非常经典的题目,剑指offer 上也有涉及本题。

方法一:线性构造+多路归并
在这里插入图片描述
在此,多路归并三个有序数组,用三个指针每次找最小值,然后每次向后移动即可。

  • 故问题关键在于,如何得到这三个有序数组?当然可以开空间预处理得到,但是有个更巧妙的方法:
  • 在这里插入图片描述

看代码会方便理解很多,也可以简单的手动模拟一下代码执行过程。看着像是一个多指针,但本质上还是一个多路归并算法。


  • 时间复杂度 O ( n ) O(n) O(n)
  • 空间复杂度 O ( 1 ) O(1) O(1)

方法二:STL堆+通用方法

  • 通用方法指的是,不论多少个质因子,都可以直接做,能够处理 力扣 313. 超级丑数 这个问题。
  • 先将 1 这个丑数放进小顶堆,每次取出堆顶元素,这个堆顶元素 *2、*3、*5 也一定是丑数,将其再放入堆即可。
  • 但是要注意,可能会重复放入,即堆中可能有重复数字。所以在堆顶元素出堆的时候,要将与它相同的全部弹出堆,也就是当前最小的。
  • 我在此处理的是用哈希set,来维护堆中元素,不让其重复插入。

代码:

方法一:线性构造+多路归并

class Solution {
public:
    int nthUglyNumber(int n) {
        vector<int> q(1, 1);
        int i = 0, j = 0, k = 0;
        while ( -- n) {
            int t = min({q[i] * 2, q[j] * 3, q[k] * 5});
            q.push_back(t);
            if (t == q[i] * 2) i ++ ;
            if (t == q[j] * 3) j ++ ;
            if (t == q[k] * 5) k ++ ;
        }
        return q.back();
    }
};

方法二:STL堆+通用方法

class Solution {
public:
    int nthUglyNumber(int n) {
        typedef long long LL;
        unordered_set<LL> S;
        priority_queue<LL, vector<LL>, greater<LL>> heap;
        heap.push(1); S.insert(1);
        LL res = 0;
        for (LL i = 0; i < n; i ++ ) {
            LL cur = heap.top(); heap.pop();
            res = cur;
            if (!S.count(cur * 2)) heap.push(cur * 2); S.insert(cur * 2);
            if (!S.count(cur * 3)) heap.push(cur * 3); S.insert(cur * 3);
            if (!S.count(cur * 5)) heap.push(cur * 5); S.insert(cur * 5);
        }

        return res;
    }
};

更好的维护重复元素的方式:来自[剑指-Offer] 49. 丑数(思维、代码优化、巧妙解法)

class Solution {
public:
    int nthUglyNumber(int n) {
        priority_queue<long, vector<long>, greater<long>> pq;
        pq.push(1);
        for (long i = 1; i < n; ++i) {
            long t = pq.top(); pq.pop();
            while (!pq.empty() && pq.top() == t) {
                t = pq.top(); pq.pop();
            }
            pq.push(t * 2);
            pq.push(t * 3);
            pq.push(t * 5);
        }
        return pq.top();
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ypuyu

如果帮助到你,可以请作者喝水~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值