0507-完美数

Raphael Liu Lv10

对于一个 正整数 ,如果它和除了它自身以外的所有 正因子 之和相等,我们称它为 「完美数」

给定一个 **整数 **n, 如果是完美数,返回 true;否则返回 false

示例 1:

**输入:** num = 28
**输出:** true
**解释:** 28 = 1 + 2 + 4 + 7 + 14
1, 2, 4, 7, 和 14 是 28 的所有正因子。

示例 2:

**输入:** num = 7
**输出:** false

提示:

  • 1 <= num <= 108

方法一:枚举

我们可以枚举 num 的所有真因子,累加所有真因子之和,记作 sum。若 sum}=\textit{num 则返回 true,否则返回 false。

在枚举时,我们只需要枚举不超过 \sqrt\textit{num 的数。这是因为如果 num 有一个大于 \sqrt\textit{num 的因数 d,那么它一定有一个小于 \sqrt\textit{num 的因数 \dfrac{\textit{num}}{d。

在枚举时,若找到了一个因数 d,那么就找到了因数 \dfrac{\textit{num}}{d。注意当 d\cdot d=\textit{num 时这两个因数相同,此时不能重复计算。

[sol1-Python3]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
class Solution:
def checkPerfectNumber(self, num: int) -> bool:
if num == 1:
return False

sum = 1
d = 2
while d * d <= num:
if num % d == 0:
sum += d
if d * d < num:
sum += num / d
d += 1
return sum == num
[sol1-C++]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
class Solution {
public:
bool checkPerfectNumber(int num) {
if (num == 1) {
return false;
}

int sum = 1;
for (int d = 2; d * d <= num; ++d) {
if (num % d == 0) {
sum += d;
if (d * d < num) {
sum += num / d;
}
}
}
return sum == num;
}
};
[sol1-Java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class Solution {
public boolean checkPerfectNumber(int num) {
if (num == 1) {
return false;
}

int sum = 1;
for (int d = 2; d * d <= num; ++d) {
if (num % d == 0) {
sum += d;
if (d * d < num) {
sum += num / d;
}
}
}
return sum == num;
}
}
[sol1-C#]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class Solution {
public bool CheckPerfectNumber(int num) {
if (num == 1) {
return false;
}

int sum = 1;
for (int d = 2; d * d <= num; ++d) {
if (num % d == 0) {
sum += d;
if (d * d < num) {
sum += num / d;
}
}
}
return sum == num;
}
}
[sol1-Golang]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
func checkPerfectNumber(num int) bool {
if num == 1 {
return false
}

sum := 1
for d := 2; d*d <= num; d++ {
if num%d == 0 {
sum += d
if d*d < num {
sum += num / d
}
}
}
return sum == num
}
[sol1-C]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
bool checkPerfectNumber(int num){
if (num == 1) {
return false;
}

int sum = 1;
for (int d = 2; d * d <= num; ++d) {
if (num % d == 0) {
sum += d;
if (d * d < num) {
sum += num / d;
}
}
}
return sum == num;
}
[sol1-JavaScript]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
var checkPerfectNumber = function(num) {
if (num === 1) {
return false;
}

let sum = 1;
for (let d = 2; d * d <= num; ++d) {
if (num % d === 0) {
sum += d;
if (d * d < num) {
sum += Math.floor(num / d);
}
}
}
return sum === num;
};

复杂度分析

  • 时间复杂度:O(\sqrt\textit{num})。
  • 空间复杂度:O(1)。

方法二:数学

根据欧几里得-欧拉定理,每个偶完全数都可以写成

2^{p-1}(2^p-1)

的形式,其中 p 为素数且 2^p-1 为素数。

由于目前奇完全数还未被发现,因此题目范围 [1,10^8] 内的完全数都可以写成上述形式。

这一共有如下 5 个:

6, 28, 496, 8128, 33550336

[sol2-Python3]
1
2
3
class Solution:
def checkPerfectNumber(self, num: int) -> bool:
return num == 6 or num == 28 or num == 496 or num == 8128 or num == 33550336
[sol2-C++]
1
2
3
4
5
6
class Solution {
public:
bool checkPerfectNumber(int num) {
return num == 6 || num == 28 || num == 496 || num == 8128 || num == 33550336;
}
};
[sol2-Java]
1
2
3
4
5
class Solution {
public boolean checkPerfectNumber(int num) {
return num == 6 || num == 28 || num == 496 || num == 8128 || num == 33550336;
}
}
[sol2-C#]
1
2
3
4
5
public class Solution {
public bool CheckPerfectNumber(int num) {
return num == 6 || num == 28 || num == 496 || num == 8128 || num == 33550336;
}
}
[sol2-Golang]
1
2
3
func checkPerfectNumber(num int) bool {
return num == 6 || num == 28 || num == 496 || num == 8128 || num == 33550336
}
[sol2-C]
1
2
3
bool checkPerfectNumber(int num){
return num == 6 || num == 28 || num == 496 || num == 8128 || num == 33550336;
}
[sol2-JavaScript]
1
2
3
var checkPerfectNumber = function(num) {
return num === 6 || num === 28 || num === 496 || num === 8128 || num === 33550336;
};

复杂度分析

  • 时间复杂度:O(1)。
  • 空间复杂度:O(1)。
 Comments
On this page
0507-完美数