欢迎访问我的GitHub

这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos

本篇概览

本篇概览

  • 这是道高频面试题,值得一看
  • 首先,这道题的难度是中等
  • 来看题目描述:
  1. 给你一个整数 n ,返回 和为 n 的完全平方数的最少数量
  2. 完全平方数 是一个整数,其值等于另一个整数的平方;换句话说,其值等于一个整数自乘的积。例如,149 16 都是完全平方数,而 3 11 不是。
  • 示例1:
  1. 输入:n = 12
  2. 输出:3
  3. 解释:12 = 4 + 4 + 4
  • 示例 2:
  1. 输入:n = 13
  2. 输出:2
  3. 解释:13 = 4 + 9
  • 提示:
  1. 1 <= n <= 104

解题思路

  • 该题的解题思路是动态规划,核心解法有两点:
  1. 数字i,可能是某个数字的平方,例如数字9是数字3的平方
  2. 数字i,如果不是某个数字的平方,该数字能用此表达式表达:i = i - j*j + j*j
  • 对于上述第二种情况,就是动态规划状态转移方程的核心啦!
  • 假设dp[i]的定义是数字i的完全平方数的最少数量,那么表达式i = i - j*j + j*j就很容易用来分析dp[i]了

  • 简单地说,就是:dp[i] = dp[i-j*j] + 1
  • 当然了,上述只是最基本的推测,不代表已经解完了,还剩一个重点:j到底是几?
  • 以10为例,10=(10-3*3) + 3*3,但是这不是唯一,还有10=(10-2*2) + 2*2,所以到底j等于几?根据题意,应该是dp[10-3*3]和dp[10-2*2]中最小的那个
  • 至此,分析完毕,可以愉快的写代码了

编码

  • 完整源码如下所示,可见,对应前面分析的j的多种可能,要取最小值
  1. class Solution {
  2. public int numSquares(int n) {
  3. // i = i-j*j + j*j - 注意这个j*j,就是完全平方数中的一个
  4. // dp[i]定义:数字i的完全平方数
  5. int[] dp = new int[n+1];
  6. dp[0] = 1;
  7. for (int i=1;i<=n;i++) {
  8. dp[i] = Integer.MAX_VALUE;
  9. for(int j=1;j*j<=i;j++) {
  10. // 如果出现i等于某个数字的平方,那么i的完全平方数就是1
  11. if (j*j==i) {
  12. dp[i] = 1;
  13. break;
  14. }
  15. // +1的意思就是j*j表示完全平方数中的一个
  16. dp[i] = Math.min(dp[i-j*j]+1, dp[i]);
  17. }
  18. }
  19. return dp[n];
  20. }
  21. }
  • 编码完成后提交,顺利AC,只是成绩很不理想,仅超过45%,如下图

反思,为啥成绩这么差?

  • 这么简单的动态规划操作,为何成绩这么落后?
  • 于是,我想到了一种可能:说不定可以作弊...
  • 理由有二
  1. 首先,这道题的输入是个数字,输出也是个数字,那就存在提前算好的可能,然后按输入返回提前算好的记过
  2. 其次,也是最关键的,就是题目要求中的那句提示,如下图,n小于等于一万,所以,我只要存一万个数字的对应关系,就行了呗:

  • 看到这里,聪明的您应该知道我要如何作弊了,没错,就是把每个数字的完全平方数算出来,改动如下图

  • 然后,运行上述代码,入参是10000,即可在控制台得到一个字符串,那就是从0到10000,每个数字的完全平方数
  • 接下来的要做的就很简单了,如下所示,用上述字符串做成一个int数组array,然后numSquares方法中就一行代码,返回入参n对应的完全平方数就行了
  1. class Solution {
  2. // 数组的值就是刚才打印出来的字符串,太长了,就不完全贴出来了
  3. private int[] array = new int [] {1,1,2,3,1,2,3,4,2,1...};
  4. public int numSquares(int n) {
  5. return array[n];
  6. }
  7. }
  • 至此,就一行代码了,相信成绩不会差了吧,运行一下试试,如下图,大跌眼镜了,一行代码也要45ms,从之前的超过45%跌落到超过22%

  • 突如其来的丢脸...

  • 好吧,让我对着这一行代码捋捋,代码太少了,很容易捋清楚,如下图

  • 找到了问题,改起来也就很容易了,如下图黄框所示,这一下,array数组在编译成class文件的时候被丢进了常量区,每次创建Solution实例的时候,不会再去创建array对象了

  • 再次提交,这一回,作弊成功,用时和内存消耗双双超过百分之九十七

  • 总的来说,动态规划是正解,如果条件允许,也能用歪门邪道作弊试试,可以开阔思路,同时取得好成绩,令人身心愉悦

欢迎关注博客园:程序员欣宸

学习路上,你不孤单,欣宸原创一路相伴...

LeetCode279:完全平方数,动态规划解法超过46%,作弊解法却超过97%的更多相关文章

  1. LeetCode 62,从动态规划想到更好的解法

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是LeetCode专题第36篇文章,我们一起来看下LeetCode的62题,Unique Paths. 题意 其实这是一道老掉牙的题目了 ...

  2. leetcode-91-解码方法(动态规划和递归两种解法)

    题目描述: 一条包含字母 A-Z 的消息通过以下方式进行了编码: 'A' -> 1 'B' -> 2 ... 'Z' -> 26 给定一个只包含数字的非空字符串,请计算解码方法的总数 ...

  3. 好!recover-binary-search-tree(难)& 两种好的空间O(n)解法 & 空间O(1)解法

    https://leetcode.com/mockinterview/session/result/xyc51it/https://leetcode.com/problems/recover-bina ...

  4. 图解leetcode279 —— 完全平方数

    每道题附带动态示意图,提供java.python两种语言答案,力求提供leetcode最优解. 描述: 给定正整数 n,找到若干个完全平方数(比如 1, 4, 9, 16, ...)使得它们的和等于  ...

  5. 98. 验证二叉搜索树 前序遍历解法以及后续遍历解法(go语言)

    leetcode题目 98. 验证二叉搜索树 前序遍历 最简洁的答案版本,由于先判断的是根节点,所以直接判断当前root的值v,是否满足大于左子树最大,小于右子树最小,然后再遍历左子树,右子树是否是这 ...

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

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

  7. leetcode279. 完全平方数

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

  8. Letter Combinations of a Phone Number:深度优先和广度优先两种解法

    Letter Combinations of a Phone Number Given a digit string, return all possible letter combinations ...

  9. LeetCode算法题-Number Complement(Java实现-五种解法)

    这是悦乐书的第240次更新,第253篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第107题(顺位题号是476).给定正整数,输出其补码数.补充策略是翻转其二进制表示的位 ...

  10. LeetCode算法题-Intersection of Two Arrays(Java实现-四种解法)

    这是悦乐书的第207次更新,第219篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第75题(顺位题号是349).给定两个数组,编写一个函数来计算它们的交集.例如: 输入: ...

随机推荐

  1. 如何使用Go中的Weighted实现资源管理

    1. 简介 本文将介绍 Go 语言中的 Weighted 并发原语,包括 Weighted 的基本使用方法.实现原理.使用注意事项等内容.能够更好地理解和应用 Weighted 来实现资源的管理,从而 ...

  2. CentOS Linux 7 配置 nginx 支持 CGI

    Nginx 本身不能执行外部程序,Nginx 处理 PHP 是通过 PHP 的 fastcgi 管理器(php-fpm)进行处理,然后 nginx 再将结果返回给用户:所以如果我们需要通过 cgi 程 ...

  3. 如何获取 C#程序 内核态线程栈

    一:背景 1. 讲故事 在这么多的案例分析中,往往会发现一些案例是卡死在线程的内核态栈上,但拿过来的dump都是用户态模式下,所以无法看到内核态栈,这就比较麻烦,需要让朋友通过其他方式生成一个蓝屏的d ...

  4. NVIDIA Maxine Video Effects SDK 編程指南 - 实践小记

    NVIDIA Maxine Video Effects SDK 編程指南 - 实践小记 本篇博客重点只说Video Effect的部分,此外还有Audio Effect的部分.还有AR部分,不在本篇范 ...

  5. 快速取模算法(Barrett Reduction)

    原理:取模运算低效的原因本质是除法运算的低效.如果能将除法变成其它运算就可以加速.具体地,将除以任意数转化成"乘一个数.除以一个 \(2^k\) "(取 \(2^{62}\) 即可 ...

  6. 在Linux环境下通过命令行执行JMeter脚本后查看响应结果的配置

    在Linux环境中进行性能测试时,我们可能会遇到一定程度的报错.如果无法打开JMeter的GUI界面,但又需要查看响应结果,可以按照以下步骤进行配置: 1. 打开JMeter的安装目录,在`bin/` ...

  7. 推荐一款C#开源的操作简单、免费的屏幕录制和GIF动画制作神器

    前言 今天要给大家推荐一款由C#语言开发且开源的操作简单.免费的屏幕录制和GIF动画制作神器:ScreenToGif . 工具介绍 ScreenToGif 是一款免费的开源屏幕录制和GIF 制作工具. ...

  8. 2023-07-16:讲一讲Kafka与RocketMQ中零拷贝技术的运用?

    2023-07-16:讲一讲Kafka与RocketMQ中零拷贝技术的运用? 答案2023-07-16: 什么是零拷贝? 零拷贝(英语: Zero-copy) 技术是指计算机执行操作时,CPU不需要先 ...

  9. 浮点指令之找main函数

    环境 vs2019 编译选项x86(32位) debug版本 float指令练习 //c++源码 #include<stdio.h> int main(int argc,char* arg ...

  10. Git:多人写作时,如何保证代码一致性

    解决方案 git add . git commit -m "message" git pull origin develop # 拉取并合并dev分支上的代码 git push