力扣第84题:完全平方数

第一部分:问题描述

给你一个整数 n ,返回 n完全平方数最少数量** 。

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

示例 1:

复制代码
输入:n = 12

输出:3 
解释:12 = 4 + 4 + 4

示例 2:

复制代码
输入:n = 13

输出:2
解释:13 = 4 + 9

第二部分:代码实现

(1)核心思想

①定义 dp[i]:表示和为 i 的完全平方数的最少数量

②状态转移:要得到 i,可以从 i - k² 加一个完全平方数 得到,所以:

因为 k 可以取很多值(1, 2, 3...),每种取值都会得到一个方案数:

  • dp[12-1] + 1 = dp[11] + 1
  • dp[12-4] + 1 = dp[8] + 1
  • dp[12-9] + 1 = dp[3] + 1

我们要从这些方案中选出数量最少 的那个,所以外面套了一个 min()

③边界条件:dp[0] = 0(和为 0 需要 0 个平方数)。

(2)完整提交代码

复制代码
class Solution:
    def numSquares(self, n: int) -> int:
        #初始化dp数组,dp[i]初始化为最坏情况(全用1的平方,总共i个)
        dp=[i for i in range(n+1)]#写到n+1是因为一会要用到dp[n]
        for i in range(1,n+1):
            #遍历所有小于等于i的平方数
            for k in range(1,int(math.sqrt(i)+1)):#sqrt是根号的意思
                square=k*k
                dp[i]=min(dp[i],dp[i-square]+1)
        return dp[n]

代码解析:

  • 初始化dp = [i for i in range(n + 1)]
    • 最坏情况:每个数都由 i 组成,所以 dp[i] = i
  • 外层循环 :遍历 1n,计算每个数的最小平方数数量。
  • 内层循环 :遍历所有小于等于 i 的平方数 ,更新 dp[i]
    • dp[i] = min(dp[i], dp[i - square] + 1):比较当前值和「用 加上 i - square 的最小数量」哪个更小。
  • 返回结果dp[n] 就是和为 n 的最少完全平方数数量。
相关推荐
无敌昊哥战神29 分钟前
【保姆级题解】力扣17. 电话号码的字母组合 (回溯算法经典入门) | Python/C/C++多语言详解
c语言·c++·python·算法·leetcode
脱氧核糖核酸__31 分钟前
LeetCode热题100——238.除了自身以外数组的乘积(题目+题解+答案)
数据结构·c++·算法·leetcode
再卷也是菜31 分钟前
算法提高篇(1)线段树(上)
数据结构·算法
py有趣34 分钟前
力扣热门100题之单词拆分
算法·leetcode
j_xxx404_2 小时前
C++算法:哈希表(简介|两数之和|判断是否互为字符重排)
数据结构·c++·算法·leetcode·蓝桥杯·力扣·散列表
Aaron15882 小时前
RFSOC+VU13P+RK3588的核心优势与应用场景分析
嵌入式硬件·算法·matlab·fpga开发·信息与通信·信号处理·基带工程
优家数科2 小时前
精准预测:基于多维用水量的滤芯寿命预警算法
算法
脱氧核糖核酸__2 小时前
LeetCode热题100——189.轮转数组(题解+答案+要点)
数据结构·c++·算法·leetcode
贾斯汀玛尔斯3 小时前
每天学一个算法-快速排序(Quick Sort)
数据结构·算法
炽烈小老头3 小时前
【每天学习一点算法 2026/04/16】逆波兰表达式求值
学习·算法