力扣 279. 完全平方数

题目说明

  • 给定正整数 n,找到若干个完全平方数(比如 1, 4, 9, 16, ...)使得它们的和等于 n。你需要让组成和的完全平方数的个数最少。

    给你一个整数 n ,返回和为 n 的完全平方数的 最少数量

    完全平方数 是一个整数,其值等于另一个整数的平方;换句话说,其值等于一个整数自乘的积。例如,14916 都是完全平方数,而 311 不是。

  • 提示:

    • 1 <= n <= 10^4

示例

示例 1:

1
2
3
输入:n = 12
输出:3
解释:12 = 4 + 4 + 4

示例 2:

1
2
3
输入:n = 13
输出:2
解释:13 = 4 + 9

笔者理解

此题是一道动态规划算法问题,在力扣题库中被定义为中等题。

解法

当笔者阅读完此题后,发现此题可以使用数学原理,还可以使用动态规划,让我们来看看具体如何实现的吧。

实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
class Solution {
/*
* 数学原理:任何一个数可以表示成不超过四个数平方的总和
*/
public int numSquares1(int n) {
// 判4:一个数最少可以拆成4个数的平方和,则这个数还满足 n = (4^a)*(8b+7)
while(n % 4 == 0) {
n /= 4;
}
if(n % 8 == 7) {
return 4;
}

// 判1:当前数是完全平方数
for(int i = 0; i * i <= n; ++i) {
if(n - i * i == 0) {
return 1;
}
}

// 判2:两个平方数的和
for(int i = 0; i * i < n; ++i) {
for(int j = 0; j * j < n; ++j) {
if(n - i * i - j * j == 0) {
return 2;
}
}
}

// 排除法:4、1、2,都不是,直接返回3
return 3;
}


/*
* 动态规划
*/
public int numSquares2(int n) {
int[] dp = new int [n + 1];

for (int i = 0; i <= n; i++) {
// 每个数最大的可能也就是全由一组成
dp[i] = i;

// 尝试小于当前j的所有与平方和的组合
// 比如dp[5]变成了dp[4] + dp[1] = 2
// 因为dp[n] = 1(n是平方数),此时采用n来组成j最优
// 所有我们枚举尝试j的最优解
for (int j = 1; i - j * j >= 0; j++) {
dp[i] = Math.min(dp[i], dp[i - j * j] + 1);
}
}

return dp[n];
}
}

时间效率和空间效率都还行,可见这两种解法还比较适合此题;

数学解法:

image.png

动态规划:

image.png

总结

本题是今天的每日一题,难度是为中等,感兴趣的朋友都可以去尝试一下,此题还有其他更多的解法,朋友们可以自己逐一尝试。