# 《程序员数学:判断2次方数》—— 除法、二进制、对数,你会用哪种方式判断?
作者:小傅哥
博客:https://bugstack.cn (opens new window)
源码:https://github.com/fuzhengwei/java-algorithms (opens new window)
沉淀、分享、成长,让自己和他人都能有所收获!😄
# 一、前言
每一个知识的索引都可能会牵扯出一片的内容
记得以前看到一个文章内容,说国外的小孩留一个很开放的作文题目,但如果想把这样一个作业写下来,则需要索引大量的资料才能完成。但在这个过程中其实会收获很多,也学会了如何学习。
其实像小傅哥去编写这样的《程序员数学》资料时,也会去横向和纵向的对比一些数学上的文章和内容,有的来自维基百科,有的来自于论文资料,现在还可以从 chatGPT (opens new window) 中探索。
另外还有一份斯坦福大学的课程资料,小傅哥把它转成 PDF 有需要的伙伴可以下载学习使用。《计算机课程资料 - 斯坦福大学 @肖恩·埃隆·安德森》 (opens new window)
# 二、判断2次方数
如果说判断一个数字是否满足2的次方,首先我会想到二进制,因为二进制的每一位都是2的次方数字。那么判断一个数字是否为2的次方可以从二进制中的数字特性下手,比如我们可以做二进制数字的判断,也就是一个数字的二进制必须只有1位为1,其他位都为0,那么这个数字就是2次方的数字。
🤔那除此之外还有其他手段吗?我们接下来就分别实现下
# 1. 整除法
代码实现
public boolean isPowerOfTwo01(int number) {
if (number < 1) return false;
int dividedNumber = number;
while (dividedNumber != 1) {
if (dividedNumber % 2 != 0) {
return false;
}
dividedNumber = dividedNumber / 2;
}
return true;
}
1
2
3
4
5
6
7
8
9
10
11
12
13
2
3
4
5
6
7
8
9
10
11
12
13
- 循环数字除以2的结果与2求模计算看余数是否为0,只要有一次不为0,那么就不是2的次方数。
# 2. 二进制
代码实现
public boolean isPowerOfTwo02(int number) {
if (number < 1) return false;
return (number & (number - 1)) == 0;
}
1
2
3
4
2
3
4
- 在斯坦福大学的计算资料中,有这么一条关于判断2的次方数的内容;
f = (v & (v - 1)) == 0;
错位进行 & 与运算,结果为0则是2次方数【过程如图】。 - 此外这里我们过滤了小于的数字,如果不过滤则需要使用;
f = v && !(v & (v - 1));
# 3. 求对数
代码实现
public boolean isPowerOfTwo03(int number) {
if (number == 0)
return false;
// log8 = log2^3 / log2 = 3
double x = Math.log(number) / Math.log(2);
// 向上和向下取整的结果判断
return (int)(Math.ceil(x)) == (int)(Math.floor(x));
}
1
2
3
4
5
6
7
8
2
3
4
5
6
7
8
- 此方式为计算2为底数的对数,如果得到的数字向上和向下取整结果一致,那么则是2的次方。
- 这里做一个简单的推到,加入 number = 8,那么 log8 = log2^3 也就是用 log2^3 与 log2 做对比。这样就能看出来一个结果3,这个3是一个整数数字,则这个 number 也就是2次方数。
# 4. 位计算
代码实现
public boolean isPowerOfTwo04(int number){
int cnt = 0;
while (number > 0) {
if ((number & 1) == 1) {
cnt++;
}
number = number >> 1;
}
return cnt == 1;
}
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
- 这个其实就是最开始说的,如果一个数字满足2次方,那么只要在二进制的转换中,它只有1位是1就可以了。
# 三、单元测试
@Test
public void test_IsPowerOfTwo(){
IsPowerOfTwo isPowerOfTwo = new IsPowerOfTwo();
System.out.println(isPowerOfTwo.isPowerOfTwo01(8));
System.out.println(isPowerOfTwo.isPowerOfTwo02(8));
System.out.println(isPowerOfTwo.isPowerOfTwo03(163));
System.out.println(isPowerOfTwo.isPowerOfTwo04(12));
}
@Test
public void test_math(){
System.out.println(Math.ceil(Math.log(8) / Math.log(2)));
System.out.println(Math.log(1));
System.out.println(Math.E);
System.out.println(Math.pow(Math.E, Math.log(2)));
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
- 在单元测试中除了验证4种判断2次方数的计算方式,也提供了关于log的计算,默认log是基于指数E的计算。读者也可以进行测试验证。a 的 x 次方 = N 那么 x = log(a)N
# 四、常见面试题
- 如何判断一个数字是2的次方数
- 在Java中怎么计算log公式