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

LeetCode第279题_完全平方数

LeetCode 第279题:完全平方数

📖 文章摘要

本文详细解析LeetCode第279题"完全平方数",这是一道经典的动态规划问题。文章提供了动态规划和数学方法两种解题思路,包含C#、Python、C++三种语言实现,配有详细的状态转移表格和性能对比。适合学习动态规划和数学优化的读者。

核心知识点: 动态规划、完全平方数、数学定理
难度等级: 中等
推荐人群: 具备基础算法知识,想要提升动态规划技能的开发者

题目描述

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

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

示例

示例 1:

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

示例 2:

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

提示

  • 1 <= n <= 10^4

解题思路

本题可以使用动态规划来解决,也可以使用数学方法。我们主要介绍这两种方法:

  1. 动态规划方法:

    • 定义dp[i]表示数字i最少需要多少个完全平方数
    • 对于每个数i,枚举所有小于等于i的完全平方数j*j
    • 状态转移方程:dp[i] = min(dp[i], dp[i - j*j] + 1)
  2. 数学方法(四平方和定理):

    • 任何正整数都可以表示为至多4个平方数之和
    • 当且仅当n≠4^k(8m+7)时,n可以表示为至多3个平方数之和

图解思路

动态规划状态转移表

数字n可用平方数dp[n]组成方式说明
1111基础情况
2121+1只能用1
3131+1+1只能用1
41,414可以直接用4
51,424+1使用4和1
121,4,934+4+4使用三个4
131,4,924+9使用4和9

数学方法分析表

情况条件结果示例说明
完全平方数n = k^2116=4^2直接是平方数
两个平方数和n = a^2 + b^2213=4+9可以分解为两个平方数之和
三个平方数和n ≠ 4^k(8m+7)312=4+4+4可以分解为三个平方数之和
四个平方数和其他情况47=4+1+1+1需要四个平方数

代码实现

C# 实现

public class Solution {public int NumSquares(int n) {// 创建dp数组,初始值设为最大可能值int[] dp = new int[n + 1];Array.Fill(dp, int.MaxValue);dp[0] = 0;// 计算所有小于等于n的完全平方数int maxSquareRoot = (int)Math.Sqrt(n);int[] squares = new int[maxSquareRoot];for (int i = 1; i <= maxSquareRoot; i++) {squares[i - 1] = i * i;}// 动态规划过程for (int i = 1; i <= n; i++) {for (int j = 0; j < maxSquareRoot && squares[j] <= i; j++) {dp[i] = Math.Min(dp[i], dp[i - squares[j]] + 1);}}return dp[n];}
}

Python 实现

class Solution:def numSquares(self, n: int) -> int:# 创建dp数组dp = [float('inf')] * (n + 1)dp[0] = 0# 计算所有小于等于n的完全平方数squares = [i * i for i in range(1, int(n ** 0.5) + 1)]# 动态规划过程for i in range(1, n + 1):for square in squares:if square > i:breakdp[i] = min(dp[i], dp[i - square] + 1)return dp[n]

C++ 实现

class Solution {
public:int numSquares(int n) {// 创建dp数组vector<int> dp(n + 1, INT_MAX);dp[0] = 0;// 计算所有小于等于n的完全平方数int maxSquareRoot = (int)sqrt(n);vector<int> squares(maxSquareRoot);for (int i = 1; i <= maxSquareRoot; i++) {squares[i - 1] = i * i;}// 动态规划过程for (int i = 1; i <= n; i++) {for (int j = 0; j < maxSquareRoot && squares[j] <= i; j++) {dp[i] = min(dp[i], dp[i - squares[j]] + 1);}}return dp[n];}
};

执行结果

C# 实现

  • 执行用时:84 ms
  • 内存消耗:27.8 MB

Python 实现

  • 执行用时:148 ms
  • 内存消耗:15.2 MB

C++ 实现

  • 执行用时:44 ms
  • 内存消耗:9.1 MB

性能对比

语言执行用时内存消耗特点
C#84 ms27.8 MB代码结构清晰,性能适中
Python148 ms15.2 MB代码最简洁,但运行较慢
C++44 ms9.1 MB性能最优,内存占用最小

代码亮点

  1. 🎯 使用动态规划优化时间复杂度
  2. 💡 预计算完全平方数数组,避免重复计算
  3. 🔍 使用Math.Min/min优化比较操作
  4. 🎨 代码结构清晰,变量命名直观

常见错误分析

  1. 🚫 未初始化dp数组为最大值
  2. 🚫 完全平方数计算范围错误
  3. 🚫 状态转移方程写错
  4. 🚫 边界条件处理不当

解法对比

解法时间复杂度空间复杂度优点缺点
动态规划O(n * sqrt(n))O(n)通用性强空间消耗大
数学方法O(sqrt(n))O(1)速度快需要数学知识
BFSO(n * sqrt(n))O(n)思路直观空间消耗大

相关题目

  • LeetCode 263. 丑数 - 简单
  • LeetCode 264. 丑数 II - 中等
  • LeetCode 313. 超级丑数 - 中等

📖 系列导航

🔥 算法专题合集 - 查看完整合集

📢 关注合集更新:点击上方合集链接,关注获取最新题解!目前已更新第279题。

💬 互动交流

感谢大家耐心阅读到这里!希望这篇题解能够帮助你更好地理解和掌握这道算法题。

如果这篇文章对你有帮助,请:

  • 👍 点个赞,让更多人看到这篇文章
  • 📁 收藏文章,方便后续查阅复习
  • 🔔 关注作者,获取更多高质量算法题解
  • 💭 评论区留言,分享你的解题思路或提出疑问

你的支持是我持续分享的动力!

💡 一起进步:算法学习路上不孤单,欢迎一起交流学习!

相关文章:

  • DeepSeek生成HTML5生命天数计算器
  • Qt实战:自定义二级选项框 | 附完整源码
  • PCL 点云旋转的轴角表示法
  • 细节/数学/滑动窗口
  • python版halcon环境配置
  • Scrapy全流程(一)
  • CLion开发Qt桌面程序_git的简单使用_小团体
  • (LeetCode 面试经典 150 题) 80. 删除有序数组中的重复项 II (双指针、栈)
  • 信贷域——信贷年审业务
  • Python漂浮的爱心
  • 闲庭信步使用SV搭建图像测试平台:第五课——使用task
  • 【Fargo】mediasoup发送2:码率分配、传输基类设计及WebRtcTransport原理
  • x1+x2=4 X1-X2=2的画图呢?
  • std::shared_ptr 的线程安全性
  • Java基础 6.21
  • cursor中定义cursor rules
  • 【Elasticsearch】运维监控:分片和节点
  • 当数据自己会说话:聚类与分类算法全景解析
  • P1220 关路灯
  • AI大模型学习之基础数学:微积分-AI大模型的数学引擎
  • 网站建设总体说明书/搜盘网
  • 菏泽做网站建设找哪家好/html静态网页制作
  • 无锡集团网站建设/北京seo报价
  • 万户做网站怎么样/网站推广代理
  • 网站上报名系统怎么做/网络服务平台
  • 哪里可以接网站开发项目做/想做游戏推广怎么找游戏公司