题目链接:5198. 丑数 III

请你帮忙设计一个程序,用来找出第 n 个丑数。

丑数是可以被 abc 整除的 正整数

示例 1:

输入:n = 3, a = 2, b = 3, c = 5

输出:4

解释:丑数序列为 2, 3, 4, 5, 6, 8, 9, 10… 其中第 3 个是 4。

示例 2:

输入:n = 4, a = 2, b = 3, c = 4

输出:6

解释:丑数序列为 2, 3, 4, 6, 8, 9, 12… 其中第 4 个是 6。

示例 3:

输入:n = 5, a = 2, b = 11, c = 13

输出:10

解释:丑数序列为 2, 4, 6, 8, 10, 11, 12, 13… 其中第 5 个是 10。

示例 4:

输入:n = 1000000000, a = 2, b = 217983653, c = 336916467

输出:1999999984

提示:
  • 1 <= n, a, b, c <= 10^9
  • 1 <= a * b * c <= 10^18
  • 本题结果在 [1, 2 * 10^9] 的范围内
前言:

这是我第一次遇见 丑数 这个概念。

如果之前见过丑数的同学可能会发现,此题的丑数定义和【丑数 - 百度百科】的定义是不同的。

此题的丑数定义:丑数是可以被 abc 整除的正整数。

题目比较坑的一点是示例 2 中的丑数序列是错的,它缺少 10。因为 10 能被 2 整除,所以 10 是丑数。

本来就没见过丑数,它还弄个错误的示例,导致写题的时候我的思路偏离。

题解:

知道丑数的定义,那么可以采用暴力搜索的方法找到第 n 个丑数。但是测试示例 4 的时候会 TLE。

因此需另寻他法。

给定一个数 n,和三个数 235,那么区间 [1, n] 有多少个丑数呢?

根据定义:我们知道 2 的倍数肯定是丑数,有多少个 2 的倍数呢?当然是 n / 2 个啦(全部向下取整)。

同理 3 的倍数也是,有 n / 3 个。

5 的倍数有 n / 5 个。

现在你会发现另一个问题,比如 62 的倍数,也是 3 的倍数。那岂不是计算了两遍?没错,确实算了两遍。因此我们需要知道 【容斥原理 - 百度百科】。说起来陌生,但是我相信大家都用过。

因此,我们可以知道区间 [1, n] 的丑数个数了。即

num=⌊na⌋+⌊nb⌋+⌊nc⌋−⌊nbc⌋−⌊nac⌋−⌊nab⌋+⌊nabc⌋num=\lfloor\frac{n}{a}\rfloor+\lfloor\frac{n}{b}\rfloor+\lfloor\frac{n}{c}\rfloor-\lfloor\frac{n}{bc}\rfloor-\lfloor\frac{n}{ac}\rfloor-\lfloor\frac{n}{ab}\rfloor+\lfloor\frac{n}{abc}\rfloornum=⌊an​⌋+⌊bn​⌋+⌊cn​⌋−⌊bcn​⌋−⌊acn​⌋−⌊abn​⌋+⌊abcn​⌋

代码表示为:num = n / a + n / b + n / c - n / bc - n / ac - n / ab + n / abc

需要注意的是,题目没有说给的三个数是互质的,因此需要计算最小公倍数和最大公约数。

然后再用【二分查找 - 百度百科】逼近第 n 个丑数 UnU_nUn​。

时间复杂度: 二分查找时间复杂度为 O(log2n)O(log2n)O(log2n)

空间复杂度: O(1)O(1)O(1)

Java:
  1. class Solution {
  2. public int nthUglyNumber( int n,
  3. int a,
  4. int b,
  5. int c) {
  6. long ab = lcm(a, b);// a,b的最小公倍数
  7. long ac = lcm(a, c);
  8. long bc = lcm(b, c);
  9. long abc = lcm(ab, c);
  10. long left = 1, right = 2000000000;
  11. while (left < right) {
  12. long mid = left + right >> 1;// 中间值,+优先级高于>>
  13. // 利用容斥原理计算区间[1, mid]的丑数
  14. long num = mid / a + mid / b + mid / c;
  15. num -= mid / bc + mid / ac + mid / ab;
  16. num += mid / abc;
  17. if (num < n) {
  18. left = mid + 1;// 取区间[mid + 1, right]
  19. } else {
  20. right = mid;// 取区间[left, mid]
  21. }
  22. }
  23. return (int) left;
  24. }
  25. // 计算最小公倍数
  26. private long lcm( long a,
  27. long b) {
  28. return a / gcd(a, b) * b;// 需要调用gcd
  29. }
  30. // 计算最大公约数
  31. private long gcd( long a,
  32. long b) {
  33. return b > 0 ? gcd(b, a % b) : a;
  34. }
  35. }

LeetCode 5198. 丑数 III(Java)容斥原理和二分查找的更多相关文章

  1. leetcode python丑数

    # Leetcode 263 丑数### 题目描述 编写一个程序判断给定的数是否为丑数. 丑数就是只包含质因数 `2, 3, 5` 的**正整数**. **示例1:** 输入: 6 输出: true ...

  2. Java学习之二分查找算法

    好久没写算法了.只记得递归方法..结果测试下爆栈了. 思路就是取范围的中间点,判断是不是要找的值,是就输出,不是就与范围的两个临界值比较大小,不断更新临界值直到找到为止,给定的集合一定是有序的. 自己 ...

  3. Java实现 LeetCode 264 丑数 II(二)

    264. 丑数 II 编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, 6, 8, ...

  4. Java实现 LeetCode 263 丑数

    263. 丑数 编写一个程序判断给定的数是否为丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例 1: 输入: 6 输出: true 解释: 6 = 2 × 3 示例 2: 输入: 8 输 ...

  5. LeetCode——264. 丑数 II

    编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, 6, 8, 9, 10, 12 ...

  6. Leetcode 264.丑数II

    丑数II 编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, 6, 8, 9, 10 ...

  7. 剑指Offer-32.丑数(C++/Java)

    题目: 把只包含质因子2.3和5的数称作丑数(Ugly Number).例如6.8都是丑数,但14不是,因为它包含质因子7. 习惯上我们把1当做是第一个丑数.求按从小到大的顺序的第N个丑数. 分析: ...

  8. leetcode 264. 丑数 II 及 313. 超级丑数

    264. 丑数 II 题目描述 编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, ...

  9. Java实现的二分查找算法

    二分查找又称折半查找,它是一种效率较高的查找方法. 折半查找的算法思想是将数列按有序化(递增或递减)排列,查找过程中采用跳跃式方式查找,即先以有序数列的中点位置为比较对象,如果要找的元素值小 于该中点 ...

随机推荐

  1. HDU 4828 小明系列故事——捉迷藏

    漂亮妹子点击就送:http://acm.hdu.edu.cn/showproblem.php?pid=4528 Time Limit: 500/200 MS (Java/Others)    Memo ...

  2. P1005 矩阵取数

    题目链接 看完题可能第一时间并没有清晰的思路.让我们一步一步的来考虑这道题目. 题目中描述操作为每次从所有的行中选取,这样做有些麻烦.仔细思考一下可以发现行与行之间互不干涉,所以我们可以对每行操作到底 ...

  3. UE4破碎物体

    1. 创建可破碎物体 首先,启用插件: 然后,选择一个模型,右键,创建可破碎物体: 2. 创建蓝图 把新创建出来的物体创建为蓝图: 击碎物体的蓝图节点: 当然,要把那个物体(图上的Destructib ...

  4. nginx location配置说明

    nginx location语法规则:location  [=|~|~*|^~]  /uri/  { … } nginx的location匹配的变量是$uri 规则优先级 = 高于 ^~ 高于 ~* ...

  5. 微信小程序之上传图片(含前后端代码例子)

    此代码示例,能够让你成功将图片上传至后端,后端做相应的处理,然后返回成功码. 前端小程序代码 index.wxml: <view class='content'> <view cla ...

  6. list删除、集合遍历删除

    public static void main(String[] args) { ArrayList<Integer> list = new ArrayList<>(); li ...

  7. js操作复选框 复选框

    //复选框点击事件 function checkAll(){ let tp=$("#tp").val(); let all=$("input[name='id']&quo ...

  8. 20190815网络与信息安全领域专项赛线上赛misc WriteUp

    目录 签到题 题目内容 使用工具 解题步骤 七代目 题目下载地址 使用工具 解题步骤 亚萨西 题目下载链接 使用工具 解题步骤 24word 题目下载链接 使用工具 解题步骤 感想 几星期前报了名却完 ...

  9. JAVA字符编码一:Unicode,GBK,GB2312,UTF-8概念基础

    第一篇:JAVA字符编码系列一:Unicode,GBK,GB2312,UTF-8概念基础 来源:holen'blog   对字符编码与Unicode,ISO 10646,UCS,UTF8,UTF16, ...

  10. docker run 中的privileged参数

    docker 应用容器 获取宿主机root权限(特殊权限-) docker run -d --name="centos7" --privileged=true centos:7 / ...