当前位置: 首页 > news >正文

【LeetCode】279.完全平方数(四种方法,不怕不会!开拓思维)

题目

链接

image-20200710145351219

题解

方法一:动态规划

思路

1,对于正整数N, 所有的解都是 N = 一个整数的平方 + 另一个整数; 直白点, N = AxA + B
2, 而B又是由 “一个整数的平方 + 另一个整数” 组成的; 那么, B = CxC + D
3,总结下就是:N = IxI + N’ 而 N’ = IxI + N’’

4, 本题要解的问题:正整数N最少由多个平方数相加;
5, 那么,N的最优解 = 1 + (N’的最优解)。而N’肯定小于N。
6, 所以本题的思路就是,对每一个N,观察1到N-1中,谁的解最小,那么N的解就是它+1.

7, 但是我们没必要1到N-1中的每一个数都去观察,因为有些组合不满足N = IxI + N’,譬如12 = 2+N’是不需要的,因为2不是某个数的平方。所以我们观察的范围要大大减小。

拿12举例,我们只能观察:
12 = 1 + 11
12 = 4 + 8
12 = 9 + 3
我们要得出3,8,11中谁的解最优,那么12的解就是它+1。

8, 我们从1到N计算, 2的解从1里找,3的解从[2,1]里找,4的解从[3,2,1]里找,依次类推,最后算到N的解即可。

数学理解

//假设最小公式值m = ƒ(n) 
//那么n的值满足下列公式 ∑(A[i] * A[i]) = n 
//令 k 为满足最小值 m 的时候,最大的平方数  。 令  d + k * k = n ;  d >= 0; 
   // 注意:一定要是满足m最小的时候的k值,一味的取最大平方数,就是贪心算法了
//得出 f(d) + f(k*k) = f(n);
//显然 f(k*k) = 1; 则  f(d) + 1 = f(n); 因为 d = n - k*k;
//则可以推出ƒ(n - k * k) + 1 = ƒ(n) ;  且 k * k <= n;

我们来理解一下动态规划方程dp[i] = Math.min(dp[i], dp[i - j * j] + 1)

    #动态方程的全写版本应该是:
    for(int j = 1;j*j<=i;j++){
		dp[i] = Math.main(dp[i],dp[i-j*j]+dp[j*j];
}
    # dp[i]:表示完全平方数和为i的 最小个数
    # 初始状态dp[i]均取最大值i,即 1+1+...+1,i个1; dp[0] = 0
    # dp[i] = min(dp[i], dp[i-j*j]+1),其中, j是平方数, j=1~k,其中k*k要保证 <= i
    # 意思就是:(完全平方数和为i的最小个数) 等于 (当前完全平方数和为i的最大个数)dp[i] 与 (完全平方数和为 i - j * j 的最小个数 + 完全平方数和为 j * j的 最小个数)的最小个数
    #   可以看到 dp[j*j] 是等于1
class Solution {
    public int numSquares(int n) {
        int[] dp = new int[n + 1]; // 默认初始化值都为0
        for (int i = 1; i <= n; i++) {
            dp[i] = i; // 最坏的情况就是每次+1
            for (int j = 1; i - j * j >= 0; j++) { 
                dp[i] = Math.min(dp[i], dp[i - j * j] + 1); // 动态转移方程
            }
        }
        return dp[n];
    }
}

方法二:贪心枚举

递归解决方法为我们理解问题提供了简洁直观的方法。我们仍然可以用递归解决这个问题。为了改进上述暴力枚举解决方案,我们可以在递归中加入贪心。我们可以将枚举重新格式化如下:

从一个数字到多个数字的组合开始,一旦我们找到一个可以组合成给定数字 n 的组合,那么我们可以说我们找到了最小的组合,因为我们贪心的从小到大的枚举组合。

为了更好的解释,我们首先定义一个名为 is_divided_by(n, count) 的函数,该函数返回一个布尔值,表示数字 n 是否可以被一个数字 count 组合,而不是像前面函数 numSquares(n) 返回组合的确切大小。

image-20200710173515393

与递归函数 numSquare(n) 不同,is_divided_by(n, count) 的递归过程可以归结为底部情况(即 count==1)更快。

下面是一个关于函数 is_divided_by(n, count) 的例子,它对 输入 n=5count=2 进行了分解。

在这里插入图片描述
通过这种重新构造的技巧,我们可以显著降低堆栈溢出的风险。

算法:

  • 首先,我们准备一个小于给定数字 n 的完全平方数列表(称为 square_nums)。
  • 在主循环中,将组合的大小(称为 count)从 1 迭代到 n,我们检查数字 n 是否可以除以组合的和,即 is_divided_by(n, count)
  • 函数 is_divided_by(n, count) 可以用递归的形式实现,汝上面所说。
  • 在最下面的例子中,我们有 count==1,我们只需检查数字 n 是否本身是一个完全平方数。可以在 square_nums 中检查,即 n \in \text{square_nums}n∈square_nums。如果 square_nums 使用的是集合数据结构,我们可以获得比 n == int(sqrt(n)) ^ 2 更快的运行时间。

关于算法的正确性,通常情况下,我们可以用反证法来证明贪心算法。这也不例外。假设我们发现 count=m 可以除以 n,并且假设在以后的迭代中存在另一个 count=p 也可以除以 n,并且这个数的组合小于找到的数,即 p<m。如果给定迭代的顺序,count = p 会在 count=m 之前被发现,因此,该算法总是能够找到组合的最小大小。

下面是一些示例实现。Python 解决方案需要大约 70ms,这比当时大约 90% 的提交要快。

class Solution {
  Set<Integer> square_nums = new HashSet<Integer>();

  protected boolean is_divided_by(int n, int count) {
    if (count == 1) {
      return square_nums.contains(n);
    }

    for (Integer square : square_nums) {
      if (is_divided_by(n - square, count - 1)) {
        return true;
      }
    }
    return false;
  }

  public int numSquares(int n) {
    this.square_nums.clear();

    for (int i = 1; i * i <= n; ++i) {
      this.square_nums.add(i * i);
    }

    int count = 1;
    for (; count <= n; ++count) {
      if (is_divided_by(n, count))
        return count;
    }
    return count;
  }
}

复杂度分析

  • 时间复杂度: O ( n h + 1 − 1 n − 1 ) = O ( n h 2 ) , \mathcal{O}( \frac{\sqrt{n}^{h+1} - 1}{\sqrt{n} - 1} ) = \mathcal{O}(n^{\frac{h}{2}}), O(n 1n h+11)=O(n2h)其中 h 是可能发生的最大递归次数。你可能会注意到,上面的公式实际上类似于计算完整 N 元数种结点数的公式。事实上,算法种的递归调用轨迹形成一个 N 元树,其中 NNsquare_nums 种的完全平方数个数。即,在最坏的情况下,我们可能要遍历整棵树才能找到最终解。
  • 空间复杂度: O ( n ) \mathcal{O}(\sqrt{n}) O(n ),我们存储了一个列表 square_nums,我们还需要额外的空间用于递归调用堆栈。但正如我们所了解的那样,调用轨迹的大小不会超过 4

方法三:贪心 + BFS(广度优先搜索)

正如上述贪心算法的复杂性分析种提到的,调用堆栈的轨迹形成一颗 N 元树,其中每个结点代表 is_divided_by(n, count) 函数的调用。基于上述想法,我们可以把原来的问题重新表述如下:

给定一个 N 元树,其中每个节点表示数字 n 的余数减去一个完全平方数的组合,我们的任务是在树中找到一个节点,该节点满足两个条件:

(1) 节点的值(即余数)也是一个完全平方数。
(2) 在满足条件(1)的所有节点中,节点和根之间的距离应该最小。

下面是这棵树的样子。

在这里插入图片描述

在前面的方法3中,由于我们执行调用的贪心策略,我们实际上是从上到下逐层构造 N 元树。我们以 BFS(广度优先搜索)的方式遍历它。在 N 元树的每一级,我们都在枚举相同大小的组合。

遍历的顺序是 BFS,而不是 DFS(深度优先搜索),这是因为在用尽固定数量的完全平方数分解数字 n 的所有可能性之前,我们不会探索任何需要更多元素的潜在组合。

算法:

  • 首先,我们准备小于给定数字 n 的完全平方数列表(即 square_nums)。
  • 然后创建 queue 遍历,该变量将保存所有剩余项在每个级别的枚举。
  • 在主循环中,我们迭代 queue 变量。在每次迭代中,我们检查余数是否是一个完全平方数。如果余数不是一个完全平方数,就用其中一个完全平方数减去它,得到一个新余数,然后将新余数添加到 next_queue 中,以进行下一级的迭代。一旦遇到一个完全平方数的余数,我们就会跳出循环,这也意味着我们找到了解。

注意:在典型的 BFS 算法中,queue 变量通常是数组或列表类型。但是,这里我们使用 set 类型,以消除同一级别中的剩余项的冗余。事实证明,这个小技巧甚至可以增加 5 倍的运行加速。

在下图中,我们以 numSquares(7) 为例说明队列的布局。

在这里插入图片描述

class Solution:
    def numSquares(self, n):

        # list of square numbers that are less than `n`
        square_nums = [i * i for i in range(1, int(n**0.5)+1)]
    
        level = 0
        queue = {n}
        while queue:
            level += 1
            #! Important: use set() instead of list() to eliminate the redundancy,
            # which would even provide a 5-times speedup, 200ms vs. 1000ms.
            next_queue = set()
            # construct the queue for the next level
            for remainder in queue:
                for square_num in square_nums:    
                    if remainder == square_num:
                        return level  # find the node!
                    elif remainder < square_num:
                        break
                    else:
                        next_queue.add(remainder - square_num)
            queue = next_queue
        return level

复杂度分析

  • 时间复杂度: O ( n h + 1 − 1 n − 1 ) = O ( n h 2 ) , \mathcal{O}( \frac{\sqrt{n}^{h+1} - 1}{\sqrt{n} - 1} ) = \mathcal{O}(n^{\frac{h}{2}}), O(n 1n h+11)=O(n2h)其中 h 是 N 元树的高度。在前面的方法三我们可以看到详细解释。
  • 空间复杂度: O ( ( n ) h ) \mathcal{O}\Big((\sqrt{n})^h\Big) O((n )h),这也是在 h 级可以出现的最大节点数。可以看到,虽然我们保留了一个完全平方数列表,但是空间的主要消耗是队列变量,它跟踪给定 N 元树级别上要访问的剩余节点。

方法四:数学运算

随着时间的推移,已经提出并证明的数学定理可以解决这个问题。在这一节中,我们将把这个问题分成几个例子。

1770 年,Joseph Louis Lagrange证明了一个定理,称为四平方和定理,也称为 Bachet 猜想,它指出每个自然数都可以表示为四个整数平方和:

p = a 0 2 + a 1 2 + a 2 2 + a 3 2 p=a_{0}^{2}+a_{1}^{2}+a_{2}^{2}+a_{3}^{2} p=a02+a12+a22+a32

其中 a_{0},a_{1},a_{2},a_{3}a0,a1,a2,a3 表示整数。

例如,3,31 可以被表示为四平方和如下:

3 = 1 2 + 1 2 + 1 2 + 0 2 31 = 5 2 + 2 2 + 1 2 + 1 2 3=1^{2}+1^{2}+1^{2}+0^{2} \qquad 31=5^{2}+2^{2}+1^{2}+1^{2} 3=12+12+12+0231=52+22+12+12

情况 1:拉格朗日四平方定理设置了问题结果的上界,即如果数 n 不能分解为较少的完全平方数,则至少可以分解为 4个完全平方数之和,即 numSquares ( n ) ≤ 4 。 \text{numSquares}(n) \le 4。 numSquares(n)4

正如我们在上面的例子中可能注意到的,数字 0 也被认为是一个完全平方数,因此我们可以认为数字 3 可以分解为 3 个或 4 个完全平方数。

然而,拉格朗日四平方定理并没有直接告诉我们用最小平方数来分解自然数。

后来,在 1797 年,Adrien Marie Legendre用他的三平方定理完成了四平方定理,证明了正整数可以表示为三个平方和的一个特殊条件:

n ≠ 4 k ( 8 m + 7 )    ⟺    n = a 0 2 + a 1 2 + a 2 2 n \ne 4^{k}(8m+7) \iff n = a_{0}^{2}+a_{1}^{2}+a_{2}^{2} n=4k(8m+7)n=a02+a12+a22

其中 k 和 m是整数。

情况 2:与四平方定理不同,Adrien-Marie-Legendre 的三平方定理给了我们一个充分必要的条件来检验这个数是否只能分解成 4 个平方。

从三平方定理看我们在第 2 种情况下得出的结论可能很难。让我们详细说明一下推论过程。

首先,三平方定理告诉我们,如果 n 的形式是$ n = 4^{k}(8m+7)$,那么 n 不能分解为 3 个平方的和。此外,我们还可以断言 n 不能分解为两个平方和,数本身也不是完全平方数。因为假设数 n 可以分解为 n = a 0 2 + a 1 2 n = a_{0}^{2}+a_{1}^{2} n=a02+a12,然后通过在表达式中添加平方数 0,即$ n = a_{0}{2}+a_{1}{2} + 0^2$,我们得到了数 n 可以分解为 3 个平方的结论,这与三平方定理相矛盾。因此,结合四平方定理,我们可以断言,如果这个数不满足三平方定理的条件,它只能分解成四个平方和。

如果这个数满足三平方定理的条件,则可以分解成三个完全平方数。但我们不知道的是,如果这个数可以分解成更少的完全平方数,即一个或两个完全平方数。

所以在我们把这个数视为底部情况(三平方定理)之前,还有两种情况需要检查,即:

情况 3.1:如果数字本身是一个完全平方数,这很容易检查,例如 n == int(sqrt(n)) ^ 2

情况 3.2:如果这个数可以分解成两个完全平方数和。不幸的是,没有任何数学定理可以帮助我们检查这个情况。我们需要使用枚举方法。

算法:

可以按照上面的例子来实现解决方案。

  • 首先,我们检查数字 n 的形式是否为$ n = 4^{k}(8m+7)$,如果是,则直接返回 4。
  • 否则,我们进一步检查这个数本身是否是一个完全平方数,或者这个数是否可以分解为两个完全平方数和。
  • 在底部的情况下,这个数可以分解为 3 个平方和,但我们也可以根据四平方定理,通过加零,把它分解为 4 个平方。但是我们被要求找出最小的平方数。
class Solution {

  protected boolean isSquare(int n) {
    int sq = (int) Math.sqrt(n);
    return n == sq * sq;
  }

  public int numSquares(int n) {
    // four-square and three-square theorems.
    while (n % 4 == 0)
      n /= 4;
    if (n % 8 == 7)
      return 4;

    if (this.isSquare(n))
      return 1;
    // enumeration to check if the number can be decomposed into sum of two squares.
    for (int i = 1; i * i <= n; ++i) {
      if (this.isSquare(n - i * i))
        return 2;
    }
    // bottom case of three-square theorem.
    return 3;
  }
}

复杂度分析

  • 时间复杂度: O ( n ) \mathcal{O}(\sqrt{n}) O(n ),在主循环中,我们检查数字是否可以分解为两个平方和,这需要$ \mathcal{O}(\sqrt{n}) $个迭代。在其他情况下,我们会在常数时间内进行检查。
  • 空间复杂度: O ( 1 ) \mathcal{O}(1) O(1),该算法消耗一个常量空间。

相关文章:

  • 【LeetCode】739.每日温度(5种方法,详细图解)
  • 【LeetCode】733.图像渲染(深度优先搜索,java实现)
  • 【LeetCode】56.合并区间(贪心算法,java实现)
  • 【LeetCode】旋转矩阵(原地选择+翻转两种方法,java实现)
  • 计算机基础(一):二进制详解
  • 计算机基础(二):压缩算法
  • 计算机基础(四):控制硬件
  • 【LeetCode】5.最长回文子串(中心扩散法,动态规划,超详细图文,java实现)
  • 【LeetCode】151.翻转字符串里的单词(三种方法,java实现)
  • 【LeetCode】28.实现strstr() (KMP超详细讲解,sunday解法等五种方法,java实现)
  • 【LeetCode】209.长度最小的子数组(滑动窗口,双指针等五种方法助你开阔思路,java实现)
  • Typora快捷键大全(含Windows和mac)!提升你的写作效率
  • Java基础(七):栈 Stack(使用方法详解)
  • Java基础(六):HashMap(使用方法详解)
  • Java基础 (三):LinkedList(含使用方法详解)
  • 2017-08-04 前端日报
  • Asm.js的简单介绍
  • es6要点
  • gf框架之分页模块(五) - 自定义分页
  • Java,console输出实时的转向GUI textbox
  • Python_网络编程
  • Python语法速览与机器学习开发环境搭建
  • TypeScript迭代器
  • Vue学习第二天
  • 规范化安全开发 KOA 手脚架
  • 模型微调
  • 微信端页面使用-webkit-box和绝对定位时,元素上移的问题
  • 线性表及其算法(java实现)
  • 移动端唤起键盘时取消position:fixed定位
  • 专访Pony.ai 楼天城:自动驾驶已经走过了“从0到1”,“规模”是行业的分水岭| 自动驾驶这十年 ...
  • ###C语言程序设计-----C语言学习(6)#
  • #NOIP 2014#day.2 T1 无限网络发射器选址
  • (2)Java 简介
  • (Java岗)秋招打卡!一本学历拿下美团、阿里、快手、米哈游offer
  • (Redis使用系列) SpirngBoot中关于Redis的值的各种方式的存储与取出 三
  • (zt)最盛行的警世狂言(爆笑)
  • (含react-draggable库以及相关BUG如何解决)固定在左上方某盒子内(如按钮)添加可拖动功能,使用react hook语法实现
  • (经验分享)作为一名普通本科计算机专业学生,我大学四年到底走了多少弯路
  • (论文阅读22/100)Learning a Deep Compact Image Representation for Visual Tracking
  • (十一)JAVA springboot ssm b2b2c多用户商城系统源码:服务网关Zuul高级篇
  • (转) Android中ViewStub组件使用
  • .NET / MSBuild 扩展编译时什么时候用 BeforeTargets / AfterTargets 什么时候用 DependsOnTargets?
  • .net core IResultFilter 的 OnResultExecuted和OnResultExecuting的区别
  • .NET Framework 3.5中序列化成JSON数据及JSON数据的反序列化,以及jQuery的调用JSON
  • .net 验证控件和javaScript的冲突问题
  • .net和jar包windows服务部署
  • .net开发引用程序集提示没有强名称的解决办法
  • @ComponentScan比较
  • @javax.ws.rs Webservice注解
  • [ C++ ] template 模板进阶 (特化,分离编译)
  • [AHOI2009]中国象棋 DP,递推,组合数
  • [Android学习笔记]ScrollView的使用
  • [bzoj 3124][sdoi 2013 省选] 直径
  • [cogs2652]秘术「天文密葬法」
  • [GN] Vue3.2 快速上手 ---- 核心语法2