当前位置: 首页 > news >正文

剑指Offer49 -- DP_贪心

1. 题目描述

丑数



2. 思路

很明显,丑数就是 2 , 3 , 5 2,3,5 2,3,5 的乘积组合。
最一开始,我竟然傻傻的 d f s dfs dfs + s e t set set 来求解,其实仔细想想, d f s dfs dfs 肯定是不行的,因为 d f s dfs dfs 会一条路走到黑。
理想情况下,我们希望每个丑数都按位置放置,而不是跳着来的,例如, 6 6 6 之后的丑数应该是 8 8 8,而不是 9 9 9
对于下一个丑数来说,他只有三种可能:

pre1 * 2;
pre2 * 3;
pre3 * 5;

pre1, pre2, pre3 可能是不同的数。我们希望每次都选取这个组合的最小值,并且刚好是下一个丑数。
那么我们就需要保证,pre1,pre2,pre3 不能大了,也不能小,并且它们都是一个个整数。
好吧,我承认我的论述有点乱,从实际的例子出发好了。
最开始,有一个特殊的丑数 1 1 1,从 1 1 1 出发,有:

1 * 2 = 2;
1 * 3 = 3;
1 * 5 = 5;

我们应该选择 2 2 2,那么,此时 p r e 1 pre1 pre1 也就是 1 ∗ 2 1*2 12 中的 1 1 1 就要编程 2 2 2 了, 2 2 2 就是 1 1 1 之后的最近的丑数。即:

2 * 2 = 4;
1 * 3 = 3;
1 * 5 = 5;

此时我们应该选择 3 3 3,那么,又变成:

2 * 2 = 4;
2 * 3 = 6;
1 * 5 = 5;

选择 4 4 4,变成;

3 * 2 = 6;
2 * 3 = 6;
1 * 5 = 5;

选择 5 5 5,变成:

3 * 2 = 6;
2 * 3 = 6;
2 * 5 = 10;

选择 6 6 6,但是此时有两个 6 6 6 同时出现,我们同时修改 p r e 2 pre2 pre2 p r e 3 pre3 pre3,因为不可以重复:

4 * 2 = 8;
3 * 3 = 9;
2 * 5 = 10;

那么,自然而然的,我们可以通过一种类似三指针的方式,求解。



3. 代码

class Solution {
public:
    int nthUglyNumber(int n) {
        int f[n + 10];
        f[0] = 1;
        int pre1 = 0, pre2 = 0, pre3 = 0;
        for(int i = 1; i <= n; i ++ ) {
            int a = f[pre1] * 2, b = f[pre2] * 3, c = f[pre3] * 5;
            int minx = min({a, b, c});
            f[i] = minx;
            // 注意下面不能都用if-else
            // 因为丑数不可以重复出现
            if(a == minx)   pre1 ++ ;
            if(b == minx)   pre2 ++ ;
            if(c == minx)   pre3 ++ ;
        }
        return f[n - 1];
    }
};

4. 思路2,贪心 + 优先队列 + set

思路就是,每次选取一个最小值 x x x ,将它发展出来的三个丑数 x ∗ 2 x*2 x2, x ∗ 3 x*3 x3, x ∗ 5 x*5 x5 放到优先队列当中。
然后,将每次从优先队列取出来的数放到哈希表中,当放入哈希表中的数的个数为 n n n 时,就表示我们取出来最小的 n n n 个丑数。
然后,就是注意去重就行了。

5. 代码

class Solution {
public:
    int nthUglyNumber(int n) {
        priority_queue<long long, vector<long long>, greater<long long>> q;
        unordered_set<int> s;
        int start = 1;
        q.push(start);
        while(q.size()) {
            auto t = q.top();   q.pop();
            if(s.count(t))  continue;   // 去重
            s.insert(t);
            if(s.size() == n)   return t;
            q.push(t * 2);
            q.push(t * 3);
            q.push(t * 5);
        }
        return -1;
    }
};

6. 总结

其实呢,这题,想到是不太好想,看别人的分析也不太容易看懂,但是看代码就一看就懂了。
其实这题,就像贪心,因为丑数只能从另一个丑数通过 pre1*2,pre2*3,pre3*5(pre都是丑数),的方式得到,因此,找出 n n n 个丑数时很容易的。
难点在于,如何保证这 n n n 个丑数恰好是最小的 n n n 个呢?贪心呗!
如果,我们希望通过 pre1*2,pre2*3,pre3*5 得到的丑数尽可能的小,那么,pre1, pre2, pre3 当然也要尽可能的小,那么,我们就每次枚举最小的 pre
如果使用过了,就要增大 pre,并且我们希望增大的幅度近可能的小,那么与 p r e pre pre 相邻的那个丑数就是最好的选择了!
另外,当我们需要增加 p r e pre pre 时,与 p r e pre pre 相邻的丑数一定存在,因为只要增大,就意味着我们找到了一个新的丑数。

相关文章:

  • 高中数学联赛模拟试题第9套几何题
  • 使用YOLOv5训练自定义数据集
  • 阿里云云效 Maven
  • 前端技术有哪些
  • Canvas渲染管线解析:从API调用到像素落地的全过程
  • 蓝桥杯省模拟赛 阶乘求值
  • QEMU源码全解析 —— 块设备虚拟化(12)
  • 线性回归 + 基础优化算法
  • docker - compose up - d`命令解释,重复运行会覆盖原有容器吗
  • 滚珠花键的预压调整怎么做?
  • 附录C SLAC匹配过程命令定义与实际抓包
  • Go 语言标准库中math模块详细功能介绍与示例
  • GRS认证的主要步骤是什么?
  • Go 语言标准库中strings和strconv详细功能介绍与示例
  • Java基础 3.28
  • K8s Calico网络介绍
  • Z字形变换
  • UI产品经理基础(六):如何解决用户的质疑?
  • Zabbix技术系列文章,第一篇——基础入门
  • 全链路压测:性能测试的流量录制和回放
  • 网站关键词重要吗/网站优化招聘
  • 简单的电影网站模板/百度浏览器官方下载
  • 企业大型网站开发需要多少钱/推广员网站
  • 导航网站cms/公众号引流推广平台
  • 日照网站优化/微信营销技巧
  • 企业简介的网站怎么做/广告软文小故事800字