每道题附带动态示意图,提供java、python两种语言答案,力求提供leetcode最优解。

描述:

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

示例 1:

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

示例 2:

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

思路:

    这道题的官方分类是【动态规划】,所以我们用动态规划的方法来解,动态规划最重要的是找到它的状态转移方程(即找出状态间的关系)。

  除了状态转移方程,我们也可以用状态转移表的方法来解题,但是状态转移表只能解维度比较低题,比如著名的0-1背包问题,影响状态转移的决策只有两种,把物品放入背包、不把物品放入背包。所以很容易就可以画出一张二维的状态转移表,但是像今天我们要解决的这种问题,假如n=12,那么影响状态转移的决策至少就有三种,取1,取4,取9,人脑很难想像出多维的状态转移表,所以这里我们采用状态转移方程的方法来解。

状态转移方程推导:

函数f(n)为求组成n的完全平方数的最小个数(就是该题),所以f(12) = 3;f(13) = 2。

我们记做f(n) = m。n可以拆分为 n = d + k*k这种形式。

比如12 = 8 + 2*2,13 = 4 + 3*3,因为无论是12还是13都是完全平方数组成的,所以一定可以转换成这种形式。

f(n) = f(d) + f(k*k),因为k*k是一个完全平方数,所以f(k*k) = 1

即f(n) = f(d) + 1,而由 n = d + k*k可得,d = n - k*k,所以上式可化为:

f(n) = f(n-k*k) + 1,(k*k < n)。

这就得出了状态转移方程:dp[i] = min(dp[i-j*j]+1, dp[i]),(j*j <= i)

这里和dp[i]取最小的原因是dp[i-j*j]+1可能不止一个值,取这些值中的最小值。

动图:

图中例子为f(5) = 2

5 = 4 + 1

实现:

java:

class Solution {
public int numSquares(int n) {
int[] dp = new int[n + 1];
for (int i = 1; i < dp.length; i++) {
dp[i] = i;
for (int j = 1; i - j * j >= 0; j++) {
dp[i] = Math.min(dp[i], dp[i - j * j]+1);
}
}
return dp[n];
}
}

结果:

python3:

class Solution:
def numSquares(self, n: int) -> int:
dp = [i for i in range(n + 1)]
for i in range(1, n + 1):
for j in range(1, n + 1):
if i - j * j >= 0:
dp[i] = min(dp[i], dp[i - j * j] + 1)
else:
break
return dp[n]

结果:

期待您的关注、推荐、收藏,同时也期待您的纠错和批评,想看leetcode的其他题,可以在博客下方留言,每周都会更新。

图解leetcode279 —— 完全平方数的更多相关文章

  1. [Swift]LeetCode279. 完全平方数 | Perfect Squares

    Given a positive integer n, find the least number of perfect square numbers (for example, 1, 4, 9, 1 ...

  2. leetcode279. 完全平方数

    给定正整数 n,找到若干个完全平方数(比如 1, 4, 9, 16, ...)使得它们的和等于 n.你需要让组成和的完全平方数的个数最少. 示例 1: 输入: n = 12输出: 3 解释: 12 = ...

  3. leetcode探索高级算法

    C++版 数组和字符串 正文 链表: 正文 树与图: 树: leetcode236. 二叉树的最近公共祖先 递归(先序) leetcode124二叉树最大路径和 递归 图: leetcode 547朋 ...

  4. Leetcode279. Perfect Squares完全平方数

    给定正整数 n,找到若干个完全平方数(比如 1, 4, 9, 16, ...)使得它们的和等于 n.你需要让组成和的完全平方数的个数最少. 示例 1: 输入: n = 12 输出: 3 解释: 12 ...

  5. 图解CSS3制作圆环形进度条的实例教程

    圆环形进度条制作的基本思想还是画出基本的弧线图形,然后CSS3中我们可以控制其旋转来串联基本图形,制造出部分消失的效果,下面就来带大家学习图解CSS3制作圆环形进度条的实例教程 首先,当有人说你能不能 ...

  6. 《图解HTTP》读书笔记

    目前国内讲解HTTP协议的书是在太少了,记忆中有两本被誉为经典的书<HTTP权威指南>与<TCP/IP详解,卷1>,但内容晦涩难懂,学习难度较大.其实,HTTP协议并不复杂,理 ...

  7. [PostgreSQL] 图解安装 PostgreSQL

    图解安装 PostgreSQL [博主]反骨仔 [原文地址]http://www.cnblogs.com/liqingwen/p/5894462.html 序 园友的一篇<Asp.Net Cor ...

  8. 图解ios程序生命周期

    图解ios程序生命周期 应用程序启动后状态有Active.Inactive.Background.Suspended.Not running这5种状态,几种状态的转换见下图: 在AppDelegate ...

  9. BZOJ 2440: [中山市选2011]完全平方数 [容斥原理 莫比乌斯函数]

    2440: [中山市选2011]完全平方数 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3028  Solved: 1460[Submit][Sta ...

随机推荐

  1. 洛谷P3366 【模板】最小生成树(kuskal)

    #include<bits/stdc++.h> using namespace std; ; ; struct node{ int cnt,fa; }f[maxn]; inline voi ...

  2. MapReduce数据流-输入

  3. LRJ-Example-06-17-Uva10562

    main() 函数中的这两行 fgets(buf[0], maxn, stdin); sscanf(buf[0], "%d", &T); 不能简单替换为 scanf(&qu ...

  4. oracle 使用显式的游标(CURSORs)

    使用隐式的游标,将会执行两次操作. 第一次检索记录, 第二次检查TOO MANY ROWS 这个exception . 而显式游标不执行第二次操作.

  5. 【Activiti工作流引擎】官方快速入门demo

    Activiti官方快速入门demo 地址: https://www.activiti.org/quick-start 0. 版本 activiti 5.22.0 JDK 1.8 1. 介绍 这个快速 ...

  6. Jenkins安装总结

    Jenkins官方文档说的安装步骤,http://jenkins.io/zh/doc/pipeline/tour/getting-started/ 相关安装资源可在官方文档下载 安装Jenkins之前 ...

  7. 用复制方式创建表 Create Table tbname as select * from user.tab where ...

    用复制方式创建表 Create Table tbname as select * from user.tab where ...

  8. python的for循环、下标和切片

    for循环的格式   for 临时变量 in 列表或者字符串:     循环满足条件时执行的代码 else:     循环不满足条件时执行的代码   例: name = "abcdef&qu ...

  9. HDU 6709“Fishing Master”(贪心+优先级队列)

    传送门 •参考资料 [1]:2019CCPC网络选拔赛 H.Fishing Master(思维+贪心) •题意 池塘里有 n 条鱼,捕捉一条鱼需要花费固定的 k 时间: 你有一个锅,每次只能煮一条鱼, ...

  10. mpvue的坑,持续更新-.-

    mpvue... 坑 怎么说呢,去github看一下,发现还是有很多问题没有解决... 不支持filter 亲,到现在还没有支持filter哦.只能用替代方法了,用computed或者渲染前先处理数据 ...