代码随想录算法训练营第四十四天| LeetCode70. 爬楼梯 (进阶)、322. 零钱兑换、279.完全平方数

一、LeetCode 70. 爬楼梯 (进阶)

题目链接/文章讲解/视频讲解:https://programmercarl.com/0070.%E7%88%AC%E6%A5%BC%E6%A2%AF%E5%AE%8C%E5%85%A8%E8%83%8C%E5%8C%85%E7%89%88%E6%9C%AC.html

状态:已解决

1.思路 

        这道题跟70.爬楼梯 - 力扣(LeetCode)很像,区别在于此题一次性能爬的台阶数不是固定的,而是题目给定的,因此就不能根据之前的递推式做了。

        那我们再来仔细看看这道题,题目给出了需要爬到的楼顶的阶数以及每次可爬的范围。那么这道题实质就是一道完全背包的题:背包容量为n,物品一共m个,且每个物品可以取无限次,问背包装入物品的排列一共有多少种。

        那么这道题就被转换成完全背包问题中的排序题了,跟前一天练的组合总和 Ⅳ-CSDN博客中的377题没有区别。

(1)确定dp数组以及下标含义:

        dp[j]: 爬到第 j 层楼梯,有dp[j]种方法。

(2)确定递推式:

        dp[i]有几种来源,dp[i - 1],dp[i - 2],dp[i - 3] 等等,即:dp[i - j]。那么递推公式为:dp[i] += dp[i - j]

(3)dp数组的初始化:

        既然递归公式是 dp[i] += dp[i - j],那么dp[0] 一定为1,dp[0]是递归中一切数值的基础所在,如果dp[0]是0的话,其他数值都是0了。

        下标非0的dp[i]初始化为0,因为dp[i]是靠dp[i-j]累计上来的,dp[i]本身为0这样才不会影响结果。

(4)确定遍历顺序:

        刚刚说了,这是一个求排列的方式,因此外层循环是容量,内层循环是物品。并且完全背包的两层循环都是从前往后遍历。

(5)举例推导dp数组:

        和刚刚的377题一致。

2.代码实现

#include<bits/stdc++.h>
using namespace std;
int main(void){
    int n,m;
    cin>>n>>m;
    vector<int> dp(n+1,0);
    dp[0] = 1;
    for(int j=0;j<=n;j++){
        for(int i=1;i<=m;i++){
            if(j >= i) dp[j] += dp[j-i];
        }
    }
    cout<<dp[n];
    return 0;
}

时间复杂度: O(n * m)

空间复杂度: O(n)

二、322. 零钱兑换

题目链接/文章讲解/视频讲解:https://programmercarl.com/0322.%E9%9B%B6%E9%92%B1%E5%85%91%E6%8D%A2.html

状态:已解决

1.思路 

        做过518. 零钱兑换 II - 力扣(LeetCode)的同学会觉得这两道题很像。确实很像,题目背景是相同的,区别在于518求的是凑钱的所有凑法,而322是求能够凑齐目标金额的最小硬币数。

(1)确定dp数组以及下标含义:

        dp[j]:凑足金额为 j 所需钱币的最少个数为dp[j]。

(2)确定递推公式:

        凑足总额为 j - coins[i] 的最少个数为dp[j - coins[i]],那么只需要加上一个钱币coins[i] 即dp[j - coins[i]] + 1就是dp[j](考虑coins[i]),因为dp[j] 要取所有 dp[j - coins[i]] + 1 中最小的,因此递推公式:dp[j] = min(dp[j - coins[i]] + 1, dp[j]);

(3)dp数组的初始化:

        首先凑齐金额为0所需的硬币数一定为0,那么其他非0下标呢?由递推公式dp[j] = min(dp[j - coins[i]] + 1, dp[j]);我们知道dp[j]是要与计算值求最小的,故为使计算值不被覆盖,初始值就应该为最大值,即:

vector<int> dp(amount+1,INT_MAX);
dp[0] = 0;

(4)确定遍历顺序:

        因为本题要求硬币的最少数量,而不是有多少种凑法,那么钱币有顺序和没有顺序都可以,都不影响钱币的最小个数。所以本题并不强调集合是组合还是排列。所以本题的两个for循环的关系是:外层for循环遍历物品,内层for遍历背包或者外层for遍历背包,内层for循环遍历物品都是可以的!

        按惯例做法,这里采用coins放在外循环,target在内循环的方式。本题钱币数量可以无限使用,那么是完全背包。故内循环是正序遍历。

(5)举例推导dp数组:

        dp[amount]为最终结果。

2.代码实现 

class Solution {
public:
    int coinChange(vector<int>& coins, int amount) {
        vector<int> dp(amount+1,INT_MAX);
        dp[0] = 0;
        for(int i=0;i<coins.size();i++){
            for(int j=coins[i];j<=amount;j++){
                if(dp[j-coins[i]] != INT_MAX)//不为初始值时才做这步
                    dp[j] = min(dp[j],dp[j-coins[i]]+1);
            }
        }
        //for(int i=0;i<=amount;i++) cout<<dp[i]<<" ";
        if(dp[amount] == INT_MAX) return -1;
        return dp[amount];
    }
};

时间复杂度:O(n * amount),n为coins长度

空间复杂度:O(amount) 

三、279.完全平方数

题目链接/文章讲解/视频讲解:https://programmercarl.com/0279.%E5%AE%8C%E5%85%A8%E5%B9%B3%E6%96%B9%E6%95%B0.html

状态:已解决

1.思路 

        换汤不换药,这道题跟上道题如出一辙:给一个容量为n的背包,求装满这个背包最少需要多少物品。物品是什么?就是一个平方数(同个数可以无限使用)。那物品的种类有多少个呢?肯定不超过sqrt(n)个!(sqrt(n)向上取整就是能够凑齐n的平方数的极限值 ),也就是说,上道题的nums[i]在这道题就是 i*i ,除此之外两道题就没有区别了。想清楚了这些,就可以开始写代码了。

(1)确定dp数组以及下标含义:

        dp[j]:和为j的完全平方数的最少数量为dp[j]。

(2)确定递推公式:

        凑足和为 j - i*i 的最少个数为dp[j - i*i],那么只需要加上一个平方数 i*i 即dp[ j - i*i ] + 1就是dp[j](考虑 i * i),因为dp[j] 要取所有 dp[j -i*i ] + 1 中最小的,因此递推公式:dp[j] = min(dp[j - i * i ] + 1, dp[j]);

(3)dp数组的初始化:

        根据题目描述,找到若干个完全平方数(比如 1, 4, 9, 16, ...),并没有从0开始,故给dp[0]=0。

        对于非0下标的dp[j],从递归公式dp[j] = min(dp[j - i * i] + 1, dp[j]) 中可以看出每次dp[j]都要选最小的,所以非0下标的dp[j]一定要初始为最大值,这样dp[j]在递推的时候才不会被初始值覆盖

(4)确定遍历顺序:

        和上题的分析是一致的

(5)举例推导dp数组:

        dp[n]为最终结果。

2.代码实现 

class Solution {
public:
    int numSquares(int n) {
        vector<int> dp(n+1,INT_MAX);
        dp[0] = 0;
        for(int i=1;i * i<=n;i++){
            for(int j=i*i;j<=n;j++){
                if(dp[j-i*i] != INT_MAX)
                    dp[j] = min(dp[j],dp[j-i*i]+1);
            }
        }
        return dp[n];
    }
};

时间复杂度: O(n * √n)

空间复杂度: O(n)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/557686.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

突破“三个九”!离子阱量子计算再创新高

如果把量子计算比作一场球赛&#xff0c;Quantinuum无疑又打了一记漂亮的好球。实际上&#xff0c;结合今年春季在量子体积、逻辑量子比特和布线问题等方面的进展&#xff0c;这个团队已经接近于完成一场完美的比赛。 3月&#xff0c;Quantinuum的研究人员证明了QCCD架构的可扩…

JavaScript 流程控制-循环

一、循环 二、 for 循环 重复执行的语句被称为循环体&#xff0c;能否继续重复执行&#xff0c;取决于循环的终止条件。 由循环体及循环的终止条件组成的语句被称为循环语句 1、语法结构 for 循环 主要用于把某些代码循环若干次&#xff0c;通常跟计数有关 for &#xff08…

springboot结合vue实现文件上传下载功能

紧接着上一次的博客&#xff0c;这次来实现一下文件(主要是图片)的上传和下载功能&#xff0c;上一次的博客如下所示&#xff1a; Springboot集成JWT token实现权限验证-CSDN博客 其实文件的上传和下载功能(后端的部分)&#xff0c;在我之前的博客就已经有写了&#xff0c;所以…

力扣经典150题第三十题:长度最小的子数组

目录 力扣经典150题解析之三十&#xff1a;长度最小的子数组1. 介绍2. 问题描述3. 示例4. 解题思路方法一&#xff1a;滑动窗口 5. 算法实现6. 复杂度分析7. 测试与验证测试用例设计测试结果分析 8. 进阶9. 总结10. 参考文献感谢阅读 力扣经典150题解析之三十&#xff1a;长度最…

重构国内游戏账号登录系统的思考和实践

本期作者 背景 账号登录系统&#xff0c;作为游戏发行平台最重要的应用之一&#xff0c;在当前的发行平台的应用架构中&#xff0c;主要承载的是用户的账号注册、登录、实名、防沉迷、隐私合规、风控等职责。合规作为企业经营的生命线&#xff0c;同时&#xff0c;账号登录作为…

数据结构系列-堆的实现

&#x1f308;个人主页&#xff1a;羽晨同学 &#x1f4ab;个人格言:“成为自己未来的主人~” 堆的实现&#xff0c;其实也就是二叉树的实现&#xff0c;我们在这里是基于数组对其进行实现的&#xff01; typedef struct Heap {HPDataType* a;int size;int capacity; }HP;…

毕业设计做一个linux操作系统怎么样?

毕业设计选择做操作系统的话&#xff0c;不太建议做的规模太大&#xff0c;你可以参考一下Linux内核的代码量&#xff0c;完全从头写的工作量还是挺大的。如果是一行一行从头写&#xff0c;学生期间&#xff0c;一学期写10000-20000行有效代码就很强了&#xff0c;而且还要学习…

【面经】2024春招-云计算后台研发工程师1(3个问题,移动TW等)

【面经】2024春招-云计算后台研发工程师1&#xff08;3个问题&#xff0c;移动&TW等&#xff09; 文章目录 岗位与面经基础1&#xff1a;数据库 & 网络&#xff08;3个问题&#xff09;基础2&#xff1a;系统 & 语法模板3&#xff1a;算法 & 项目&#xff08;移…

探索人工智能绘图的奇妙世界

探索人工智能绘图的奇妙世界 人工智能绘图的基本原理机器之美&#xff1a;AI绘图作品AI绘图对艺术创作的影响未来展望与挑战图书推荐&#x1f449;AI绘画教程&#xff1a;Midjourney使用方法与技巧从入门到精通内容简介获取方式&#x1f449;搜索之道&#xff1a;信息素养与终身…

Timelapse - 2024.04.09 -Win

阅读须知&#xff1a; 探索者安全团队技术文章仅供参考,未经授权请勿利用文章中的技术资料对任何计算机系统进行入侵操作,由于传播、利用本公众号所提供的技术和信息而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者 本人负责&#xff0c;作者不为此承担任何责任,如…

centos6.5重启docker容器死机问题

概述 近期在整理服务问题&#xff0c;使用docker容器重新部署服务。 过程中有不少坑&#xff0c;主要是系统配置和系统版本的问题。 环境 CentOS release 6.5 (Final) docker version 1.7.1 问题现象 使用restart命令重启docker容器&#xff0c;系统突然卡死&#xff0c…

protobuf抓包,读包

protobuf抓包 有时候会遇到使用protobuf协议的http请求, 而protobuf封包后的二进制几乎不可读, 如何调试呢 protobuf就是类似一个json的数据传输协议, 相比json更快, 体积更小; 缺点就是不可读 Content-Type: application/x-protobuf数据大概是下面这样的(浏览器开发者工具 自…

(十八)C++自制植物大战僵尸游戏的游戏暂停实现

植物大战僵尸游戏开发教程专栏地址http://t.csdnimg.cn/uzrnw 游戏暂停 当玩家遇到突发事件&#xff0c;可以通过暂停功能暂停游戏&#xff0c;以便及时处理问题。在激烈的游戏中&#xff0c;玩家可能需要暂停游戏来进行策略调整。此外&#xff0c;长时间的游戏对战可能会让玩…

爬取微博评论数据

# -*- coding: utf-8 -*- import requests #用于发送请求并且拿到源代码 from bs4 import BeautifulSoup #用于解析数据 1.找到数据源地址并且分析链接 2.发送请求并且拿到数据 3.在拿到的数据中解析出需要的数据 4.存储数据 headers { "User-Agent": "…

ubunt18.04安装ROS2

本文无废话&#xff0c;实现了ubunt18.04 下ros2的安装&#xff0c;并且同时兼容ros和ros2 如果想完ros&#xff08;1&#xff09;的&#xff0c;请参考我的前一篇文章&#xff1a;ubunt18.04安装ROS避坑指南 参考&#xff1a; https://blog.csdn.net/cau_weiyuhu/article/deta…

Ubuntu20.04版本命令行设置挂载磁盘,并设置开机自动挂载

最近部署应用 系统是Ubuntu20.4版本的Linux系统&#xff0c;加了数据盘&#xff0c;需要格式化后挂载&#xff0c;记录下&#xff1a; Linux 数据盘挂载(采用 parted 分区工具)-格式化为 ext4 1. 初始化 Linux 数据盘 挂载数据盘后或者随实例创建时一并创建的数据盘&#xff…

【数学建模】最优旅游城市的选择问题:层次分析模型(含MATLAB代码)

层次分析法&#xff08;The analytic hierarachy process&#xff0c;简称AHP&#xff09;是一种常用的决策分析方法&#xff0c;其基本思路是将复杂问题分解为多个组成部分&#xff0c;然后对这些部分进行逐一评估和比较&#xff0c;最后得出最优解决方案。&#xff08;例如&a…

Linux 5.10 Pstore 学习之(二) 原理学习

目录 编译框架模块初始化pstore子系统ramoops模块初始化实例化注册回调数据结构 pstore_blk模块pstore_zone模块 测试扩展调试 编译框架 目标结构 linux_5.10/fs/pstore/ ├── blk.c ├── ftrace.c ├── inode.c // 核心1 ├── internal.h ├── Kconfig ├── …

Vitis HLS 学习笔记--scal 函数-探究

目录 1. Vitis HLS重器-Vitis_Libraries 2. 初识scal() 3. 函数具体实现 3.1 变量命名规则 3.2 t_ParEntries解释 3.3 流类型详解 3.4 双重循环 4. 总结 1. Vitis HLS重器-Vitis_Libraries 在深入探索Vitis HLS&#xff08;High-Level Synthesis&#xff09;的旅程中&…

了解 containerd 中的 snapshotter,先从 native 开始

本文内容节选自 《containerd 原理剖析与实战》&#xff0c;本书正参加限时优惠内购&#xff0c;点击阅读原文&#xff0c;限时 69.9 元购买。 上一篇文章《一文了解 containerd 中的 snapshot》中&#xff0c;介绍了containerd 的 snapshot 机制&#xff0c;了解到 containerd…