Write a program to find the n-th ugly number.

Ugly numbers are positive numbers whose prime factors only include 2, 3, 5.

Example:

Input: n = 10
Output: 12
Explanation: 1, 2, 3, 4, 5, 6, 8, 9, 10, 12 is the sequence of the first 10 ugly numbers.

Note:

  1. 1 is typically treated as an ugly number.
  2. n does not exceed 1690.

Hint:

  1. The naive approach is to call isUgly for every number until you reach the nth one. Most numbers are not ugly. Try to focus your effort on generating only the ugly ones.
  2. An ugly number must be multiplied by either 2, 3, or 5 from a smaller ugly number.
  3. The key is how to maintain the order of the ugly numbers. Try a similar approach of merging from three sorted lists: L1, L2, and L3.
  4. Assume you have Uk, the kth ugly number. Then Uk+1 must be Min(L1 * 2, L2 * 3, L3 * 5).

这道题是之前那道 Ugly Number 的拓展,这里让找到第n个丑陋数,还好题目中给了很多提示,基本上相当于告诉我们解法了,根据提示中的信息,丑陋数序列可以拆分为下面3个子列表:

(1) 1x2,  2x2, 2x2, 3x2, 3x2, 4x2, 5x2...
(2) 1x3,  1x3, 2x3, 2x3, 2x3, 3x3, 3x3...
(3) 1x5,  1x5, 1x5, 1x5, 2x5, 2x5, 2x5...

仔细观察上述三个列表,可以发现每个子列表都是一个丑陋数分别乘以 2,3,5,而要求的丑陋数就是从已经生成的序列中取出来的,每次都从三个列表中取出当前最小的那个加入序列,请参见代码如下:

解法一:

class Solution {
public:
int nthUglyNumber(int n) {
vector<int> res(, );
int i2 = , i3 = , i5 = ;
while (res.size() < n) {
int m2 = res[i2] * , m3 = res[i3] * , m5 = res[i5] * ;
int mn = min(m2, min(m3, m5));
if (mn == m2) ++i2;
if (mn == m3) ++i3;
if (mn == m5) ++i5;
res.push_back(mn);
}
return res.back();
}
};

我们也可以使用最小堆来做,首先放进去一个1,然后从1遍历到n,每次取出堆顶元素,为了确保没有重复数字,进行一次 while 循环,将此时和堆顶元素相同的都取出来,然后分别将这个取出的数字乘以 2,3,5,并分别加入最小堆。这样最终 for 循环退出后,堆顶元素就是所求的第n个丑陋数,参见代码如下:

解法二:

class Solution {
public:
int nthUglyNumber(int n) {
priority_queue<long, vector<long>, greater<long>> pq;
pq.push();
for (long i = ; i < n; ++i) {
long t = pq.top(); pq.pop();
while (!pq.empty() && pq.top() == t) {
t = pq.top(); pq.pop();
}
pq.push(t * );
pq.push(t * );
pq.push(t * );
}
return pq.top();
}
};

Github 同步地址:

https://github.com/grandyang/leetcode/issues/264

类似题目:

Super Ugly Number

Ugly Number

Happy Number

Count Primes

Merge k Sorted Lists

Perfect Squares

参考资料:

https://leetcode.com/problems/ugly-number-ii/

https://leetcode.com/problems/ugly-number-ii/discuss/69372/Java-solution-using-PriorityQueue

https://leetcode.com/problems/ugly-number-ii/discuss/69364/My-16ms-C%2B%2B-DP-solution-with-short-explanation

https://leetcode.com/problems/ugly-number-ii/discuss/69368/Elegant-C%2B%2B-Solution-O(N)-space-time-with-detailed-explanation.

LeetCode All in One 题目讲解汇总(持续更新中...)

最新文章

  1. Ionic2学习笔记(10):扫描二维码
  2. NetSuite Chinese Finance Reports
  3. Python爬虫Scrapy框架入门(0)
  4. CORS(跨域资源共享)
  5. C++浅析——返回对象的函数
  6. Codeforces Round #250 (Div. 2) C、The Child and Toy
  7. iOS开发——项目实战总结&amp;带你看看Objective-C的精髓
  8. Hibernate Validator
  9. 【log4net】配置
  10. 南阳OJ----Binary String Matching
  11. iOS 9 适配需要注意的问题
  12. 《神经网络和深度学习》系列文章十二:Hadamard积,s⊙t
  13. Ubuntu 13.10 Mono安装历程
  14. Nginx配置文件常用部分详解
  15. websocket(一)--握手
  16. Android Studio 全局内替换字符串
  17. react-native 新手爬坑经历(unable to load script from assets 和could not connect to development server.)
  18. 54. Spiral Matrix以螺旋顺序输出数组
  19. lerna import &amp;&amp; add 使用&amp;&amp;常见问题解决
  20. egg 官方文档之:框架扩展(Application、Context、Request、Response、Helper的访问方式及扩展)

热门文章

  1. vs2017离线包下载安装并且不占用C盘空间使用教程
  2. Java代码开发之《异常日志》
  3. MySQL5.7安装脚本
  4. DVWA-CSRF学习笔记
  5. ABAP 新语法-实例讲解
  6. 动态ALV表实例-移动类型汇总
  7. ASP.NET Core 3.0 使用AspectCore-Framework实现AOP
  8. CSS 基础面试题
  9. xml路径错误无法打包
  10. Codeforces Round #303 (Div. 2)(CF545) E Paths and Trees(最短路+贪心)