素数分解
每一个数都可以分解成素数的乘积,例如 84 = 22 * 31 * 50 * 71 * 110 * 130 * 170 * …
整除
令 x = 2m0 * 3m1 * 5m2 * 7m3 * 11m4 * …
令 y = 2n0 * 3n1 * 5n2 * 7n3 * 11n4 * …
如果 x 整除 y(y mod x == 0),则对于所有 i,mi <= ni。
最大公约数最小公倍数
x 和 y 的最大公约数为:gcd(x,y) = 2min(m0,n0) * 3min(m1,n1) * 5min(m2,n2) * …
x 和 y 的最小公倍数为:lcm(x,y) = 2max(m0,n0) * 3max(m1,n1) * 5max(m2,n2) * …
1. 生成素数序列
204. Count Primes (Easy)
埃拉托斯特尼筛法在每次找到一个素数时,将能被素数整除的数排除掉。
1 | public int countPrimes(int n) { |
2. 最大公约数
1 | int gcd(int a, int b) { |
最小公倍数为两数的乘积除以最大公约数。
1 | int lcm(int a, int b) { |
3. 使用位操作和减法求解最大公约数
对于 a 和 b 的最大公约数 f(a, b),有:
- 如果 a 和 b 均为偶数,f(a, b) = 2*f(a/2, b/2);
- 如果 a 是偶数 b 是奇数,f(a, b) = f(a/2, b);
- 如果 b 是偶数 a 是奇数,f(a, b) = f(a, b/2);
- 如果 a 和 b 均为奇数,f(a, b) = f(b, a-b);
乘 2 和除 2 都可以转换为移位操作。
1 | public int gcd(int a, int b) { |
进制转换
1. 7 进制
504. Base 7 (Easy)
1 | public String convertToBase7(int num) { |
Java 中 static String toString(int num, int radix) 可以将一个整数转换为 radix 进制表示的字符串。
1 | public String convertToBase7(int num) { |
2. 16 进制
405. Convert a Number to Hexadecimal (Easy)
1 | Input: |
负数要用它的补码形式。
1 | public String toHex(int num) { |
3. 26 进制
168. Excel Sheet Column Title (Easy)
1 | 1 -> A |
因为是从 1 开始计算的,而不是从 0 开始,因此需要对 n 执行 -1 操作。
1 | public String convertToTitle(int n) { |
阶乘
1. 统计阶乘尾部有多少个 0
172. Factorial Trailing Zeroes (Easy)
尾部的 0 由 2 * 5 得来,2 的数量明显多于 5 的数量,因此只要统计有多少个 5 即可。
对于一个数 N,它所包含 5 的个数为:N/5 + N/52 + N/53 + …,其中 N/5 表示不大于 N 的数中 5 的倍数贡献一个 5,N/52 表示不大于 N 的数中 52 的倍数再贡献一个 5 …。
1 | public int trailingZeroes(int n) { |
如果统计的是 N! 的二进制表示中最低位 1 的位置,只要统计有多少个 2 即可,该题目出自 编程之美:2.2 。和求解有多少个 5 一样,2 的个数为 N/2 + N/22 + N/23 + …
字符串加法减法
1. 二进制加法
67. Add Binary (Easy)
1 | a = "11" |
1 | public String addBinary(String a, String b) { |
2. 字符串加法
415. Add Strings (Easy)
字符串的值为非负整数。
1 | public String addStrings(String num1, String num2) { |
相遇问题
1. 改变数组元素使所有的数组元素都相等
462. Minimum Moves to Equal Array Elements II (Medium)
1 | Input: |
每次可以对一个数组元素加一或者减一,求最小的改变次数。
这是个典型的相遇问题,移动距离最小的方式是所有元素都移动到中位数。理由如下:
设 m 为中位数。a 和 b 是 m 两边的两个元素,且 b > a。要使 a 和 b 相等,它们总共移动的次数为 b - a,这个值等于 (b - m) + (m - a),也就是把这两个数移动到中位数的移动次数。
设数组长度为 N,则可以找到 N/2 对 a 和 b 的组合,使它们都移动到 m 的位置。
解法 1
先排序,时间复杂度:O(NlogN)
1 | public int minMoves2(int[] nums) { |
解法 2
使用快速选择找到中位数,时间复杂度 O(N)
1 | public int minMoves2(int[] nums) { |
多数投票问题
1. 数组中出现次数多于 n / 2 的元素
169. Majority Element (Easy)
先对数组排序,最中间那个数出现次数一定多于 n / 2。
1 | public int majorityElement(int[] nums) { |
可以利用 Boyer-Moore Majority Vote Algorithm 来解决这个问题,使得时间复杂度为 O(N)。可以这么理解该算法:使用 cnt 来统计一个元素出现的次数,当遍历到的元素和统计元素不相等时,令 cnt–。如果前面查找了 i 个元素,且 cnt == 0,说明前 i 个元素没有 majority,或者有 majority,但是出现的次数少于 i / 2,因为如果多于 i / 2 的话 cnt 就一定不会为 0。此时剩下的 n - i 个元素中,majority 的数目依然多于 (n - i) / 2,因此继续查找就能找出 majority。
1 | public int majorityElement(int[] nums) { |
其它
1. 平方数
367. Valid Perfect Square (Easy)
1 | Input: 16 |
平方序列:1,4,9,16,..
间隔:3,5,7,…
间隔为等差数列,使用这个特性可以得到从 1 开始的平方序列。
1 | public boolean isPerfectSquare(int num) { |
2. 3 的 n 次方
326. Power of Three (Easy)
1 | public boolean isPowerOfThree(int n) { |
3. 乘积数组
238. Product of Array Except Self (Medium)
1 | For example, given [1,2,3,4], return [24,12,8,6]. |
给定一个数组,创建一个新数组,新数组的每个元素为原始数组中除了该位置上的元素之外所有元素的乘积。
要求时间复杂度为 O(N),并且不能使用除法。
1 | public int[] productExceptSelf(int[] nums) { |
4. 找出数组中的乘积最大的三个数
628. Maximum Product of Three Numbers (Easy)
1 | Input: [1,2,3,4] |
1 | public int maximumProduct(int[] nums) { |
- 本文作者: LHS
- 本文链接: https:/LiuHuAshen.github.io/2020/10/22/leetcode题解——数学/
- 版权声明: 本博客所有文章除特别声明外,均采用 MIT 许可协议。转载请注明出处!