LeetCode 5198. 丑数 III(Java)容斥原理和二分查找
题目链接:5198. 丑数 III
请你帮忙设计一个程序,用来找出第 n 个丑数。
丑数是可以被 a 或 b 或 c 整除的 正整数。
示例 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^91 <= a * b * c <= 10^18- 本题结果在
[1, 2 * 10^9]的范围内
前言:
这是我第一次遇见 丑数 这个概念。
如果之前见过丑数的同学可能会发现,此题的丑数定义和【丑数 - 百度百科】的定义是不同的。
此题的丑数定义:丑数是可以被 a 或 b 或 c 整除的正整数。
题目比较坑的一点是示例 2 中的丑数序列是错的,它缺少 10。因为 10 能被 2 整除,所以 10 是丑数。
本来就没见过丑数,它还弄个错误的示例,导致写题的时候我的思路偏离。
题解:
知道丑数的定义,那么可以采用暴力搜索的方法找到第 n 个丑数。但是测试示例 4 的时候会 TLE。
因此需另寻他法。
给定一个数 n,和三个数 2,3,5,那么区间 [1, n] 有多少个丑数呢?
根据定义:我们知道 2 的倍数肯定是丑数,有多少个 2 的倍数呢?当然是 n / 2 个啦(全部向下取整)。
同理 3 的倍数也是,有 n / 3 个。
5 的倍数有 n / 5 个。
现在你会发现另一个问题,比如 6 是 2 的倍数,也是 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:
class Solution {
public int nthUglyNumber( int n,
int a,
int b,
int c) {
long ab = lcm(a, b);// a,b的最小公倍数
long ac = lcm(a, c);
long bc = lcm(b, c);
long abc = lcm(ab, c);
long left = 1, right = 2000000000;
while (left < right) {
long mid = left + right >> 1;// 中间值,+优先级高于>>
// 利用容斥原理计算区间[1, mid]的丑数
long num = mid / a + mid / b + mid / c;
num -= mid / bc + mid / ac + mid / ab;
num += mid / abc;
if (num < n) {
left = mid + 1;// 取区间[mid + 1, right]
} else {
right = mid;// 取区间[left, mid]
}
}
return (int) left;
}
// 计算最小公倍数
private long lcm( long a,
long b) {
return a / gcd(a, b) * b;// 需要调用gcd
}
// 计算最大公约数
private long gcd( long a,
long b) {
return b > 0 ? gcd(b, a % b) : a;
}
}
LeetCode 5198. 丑数 III(Java)容斥原理和二分查找的更多相关文章
- leetcode python丑数
# Leetcode 263 丑数### 题目描述 编写一个程序判断给定的数是否为丑数. 丑数就是只包含质因数 `2, 3, 5` 的**正整数**. **示例1:** 输入: 6 输出: true ...
- Java学习之二分查找算法
好久没写算法了.只记得递归方法..结果测试下爆栈了. 思路就是取范围的中间点,判断是不是要找的值,是就输出,不是就与范围的两个临界值比较大小,不断更新临界值直到找到为止,给定的集合一定是有序的. 自己 ...
- Java实现 LeetCode 264 丑数 II(二)
264. 丑数 II 编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, 6, 8, ...
- Java实现 LeetCode 263 丑数
263. 丑数 编写一个程序判断给定的数是否为丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例 1: 输入: 6 输出: true 解释: 6 = 2 × 3 示例 2: 输入: 8 输 ...
- LeetCode——264. 丑数 II
编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, 6, 8, 9, 10, 12 ...
- Leetcode 264.丑数II
丑数II 编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, 6, 8, 9, 10 ...
- 剑指Offer-32.丑数(C++/Java)
题目: 把只包含质因子2.3和5的数称作丑数(Ugly Number).例如6.8都是丑数,但14不是,因为它包含质因子7. 习惯上我们把1当做是第一个丑数.求按从小到大的顺序的第N个丑数. 分析: ...
- leetcode 264. 丑数 II 及 313. 超级丑数
264. 丑数 II 题目描述 编写一个程序,找出第 n 个丑数. 丑数就是只包含质因数 2, 3, 5 的正整数. 示例: 输入: n = 10 输出: 12 解释: 1, 2, 3, 4, 5, ...
- Java实现的二分查找算法
二分查找又称折半查找,它是一种效率较高的查找方法. 折半查找的算法思想是将数列按有序化(递增或递减)排列,查找过程中采用跳跃式方式查找,即先以有序数列的中点位置为比较对象,如果要找的元素值小 于该中点 ...
随机推荐
- 洛谷P2068 统计和题解
题目描述 给定一个长度为n(n<=100000),初始值都为0的序列,x(x<=10000)次的修改某些位置上的数字,每次加上一个数,然后提出y (y<=10000)个问题,求每段区 ...
- JPA批量插入优化
遇到一个需求是excel数据导入,一次大概会批量插入几万的数据.写完一测奇慢无比. 于是开始打日志,分析代码,发现是插入数据库的时候耗时很长,发现是spring data jpa的原因. 翻看jpa的 ...
- Hash总结
算法介绍 这个没什么好说的,就是一段比较简单的代码,具体的话要看题目. 自然溢出 这个是比较好用而且容易被卡的一种Hash方式. 用\(2^{64}\)当模数然后做\(Hash\),常数比较小. 单模 ...
- Nexus OSS私服仓库的备份与迁移
背景 在上一篇博客 [Maven学习]Nexus OSS私服仓库的安装和配置 中,我们已经在机房搭建好了新的Nexus OSS私服仓库.下面是两个版本的Nexus OSS私服仓库的对比图. 老的Nex ...
- cgdsR 下载TCGA数据
TCGA 的数据可以在5个组织机构获取,它们都提供了类似的接口来供用户下载数据. cgdsR 包是cBioPortal 提供的R包 http://www.cbioportal.org/rmatlab ...
- mercurial branch name use integer as a name
问题:mercurial branch name use integer as a name 解决: 到安装目录下找到mercurial/scmutil.py文件(我的:/usr/local/Cell ...
- Android Studio 屏幕方向以及UI界面状态的保存
package com.example.orientation; import android.os.Bundle; import android.util.Log; import android.v ...
- kafka如何实现高并发存储-如何找到一条需要消费的数据(阿里)
阿里太注重原理了:阿里问kafka如何实现高并发存储-如何找到一条需要消费的数据,kafka用了稀疏索引的方式,使用了二分查找法,其实很多索引都是二分查找法 二分查找法的时间复杂度:O(logn) ...
- ASP.NET,C#后台调用前台javascript的五种方法
C#后台调用前台javascript的五种方法 由于项目需要,用到其他项目组用VC开发的组件,在web后台代码无法访问这个组件,所以只好通过后台调用前台的javascript,从而操作这个组件.在网上 ...
- Vue 项目中断点没有跳转到指定源码的问题
将配置中 devtool 改为以下即可. devtool: 'source-map' 如果是在 vue-cli 2.x ,那么就在 webpack.dev.config.js 中 如果是 vue-cl ...