代码随想录 day 32 动态规划

news2025/1/23 1:05:38

第九章 动态规划part01

今天正式开始动态规划!

理论基础

无论大家之前对动态规划学到什么程度,一定要先看 我讲的 动态规划理论基础。
如果没做过动态规划的题目,看我讲的理论基础,会有感觉 是不是简单题想复杂了?
其实并没有,我讲的理论基础内容,在动规章节所有题目都有运用,所以很重要!
如果做过动态规划题目的录友,看我的理论基础 就会感同身受了。
https://programmercarl.com/%E5%8A%A8%E6%80%81%E8%A7%84%E5%88%92%E7%90%86%E8%AE%BA%E5%9F%BA%E7%A1%80.html
视频:https://www.bilibili.com/video/BV13Q4y197Wg

509. 斐波那契数

很简单的动规入门题,但简单题使用来掌握方法论的,还是要有动规五部曲来分析。
https://programmercarl.com/0509.%E6%96%90%E6%B3%A2%E9%82%A3%E5%A5%91%E6%95%B0.html
视频:https://www.bilibili.com/video/BV1f5411K7mo

70. 爬楼梯

本题大家先自己想一想, 之后会发现,和 斐波那契数 有点关系。
https://programmercarl.com/0070.%E7%88%AC%E6%A5%BC%E6%A2%AF.html
视频:https://www.bilibili.com/video/BV17h411h7UH

746. 使用最小花费爬楼梯

这道题目力扣改了题目描述了,现在的题目描述清晰很多,相当于明确说 第一步是不用花费的。
更改题目描述之后,相当于是 文章中 「拓展」的解法
https://programmercarl.com/0746.%E4%BD%BF%E7%94%A8%E6%9C%80%E5%B0%8F%E8%8A%B1%E8%B4%B9%E7%88%AC%E6%A5%BC%E6%A2%AF.html
视频讲解:https://www.bilibili.com/video/BV16G411c7yZ

动规五步曲 你给我记住了,用几道简单题目来掌握方法论
  1. 确定dp数组(dp table)以及下标的含义
  2. 确定递推公式
  3. dp数组如何初始化
  4. 确定遍历顺序
    比较考究,01 先遍历背包,后遍历物品。
    4.1排列和组合的遍历顺序是不相同的。
    4.1.1 排列:背包在外 物品在内。(322)
    4.1.2 组合:物品在外,背包在内。(518)
  5. 举例推导dp数组
    (出现问题,打印dp数组,检查是否有问题,检验1 2 3 4 步骤)

509. 斐波那契数

题目链接

https://leetcode.cn/problems/fibonacci-number/description/

解题思路

用一道简单题目来掌握方法论,动规五步曲
1.确定dp数组和下标的含义
dp[i] 解读
下标i表示第i个斐波那契数,dp[i] 表示第i个斐波那契数的值
2.确定递推公式
这题简单,题目已经告诉了我们递推公式是什么
dp[i]=dp[i-1] + dp[i-2]
3.dp数组如何初始化
这题简单,题目已经告诉我们了
dp[0]=0 ,dp[1]=1
4.确定遍历顺序
我们发现d[i]是从dp[i-1]和dp[i-2]得到的,那么我们就要从前向后遍历才能得到dp[i]的结果
5.打印dp数组
主要用来debug的,如何验证我们写的代码没有问题呢,就是要把dp数组打印出来验证和我们想象的数组是否一样

code

class Solution {
    //时间复杂度O(n) 空间复杂度O(n)
    public int fib1(int n) {
        if(n<=1){
            return n;
        }
        //1.确定dp数组以及下标的含义
        //dp[i]的定义为:第i个数的斐波那契数值是dp[i]
        //2.确定递推公式
        //状态转移方程 dp[i] = dp[i - 1] + dp[i - 2];
        //3.dp数组初始化
        int[] dp=new int[n+1];
        dp[0]=0;
        dp[1]=1;
        //4.确定遍历顺序
        for(int i=2;i<=n;i++){
            dp[i]=dp[i-1]+dp[i-2];
             
        }
        //按照这个递推公式dp[i] = dp[i - 1] + dp[i - 2],我们来推导一下,当n为10的时候,dp数组应该是如下的数列:
        //0 1 1 2 3 5 8 13 21 34 55
        //如果代码写出来,发现结果不对,就把dp数组打印出来看看和我们推导的数列是不是一致的。
        //5.打印dp数组
        //System.out.println(Arrays.toString(dp));
        return dp[n];
    }

}

优化空间复杂度,只用长度为2的dp数组

    //时间O(n) 优化为空间(O)1
    public int fib(int n) {
        if(n<=1){
            return n;
        }
        int[] dp=new int[2];
        dp[0]=0;
        dp[1]=1;
        for(int i=2;i<=n;i++){
            int sum=dp[0]+dp[1];
            dp[0]=dp[1];
            dp[1]=sum;
             
        }
        return dp[1];
    }

递归写法 非常耗时

  • 时间复杂度:O(2^n) 空间复杂度:O(n),算上了编程语言中实现递归的系统栈所占空间
  • 递归写法时间复杂度非常高 画出来是一个树,一棵深度(按根节点深度为1)为k的二叉树最多可以有 2^k - 1 个节点 O(2^n)

好好看看这几篇文章分析递归的时间复杂度
https://programmercarl.com/%E5%89%8D%E5%BA%8F/%E9%80%92%E5%BD%92%E7%AE%97%E6%B3%95%E7%9A%84%E6%97%B6%E9%97%B4%E4%B8%8E%E7%A9%BA%E9%97%B4%E5%A4%8D%E6%9D%82%E5%BA%A6%E5%88%86%E6%9E%90.html
x的n次方
https://programmercarl.com/%E5%89%8D%E5%BA%8F/%E9%80%9A%E8%BF%87%E4%B8%80%E9%81%93%E9%9D%A2%E8%AF%95%E9%A2%98%E7%9B%AE%EF%BC%8C%E8%AE%B2%E4%B8%80%E8%AE%B2%E9%80%92%E5%BD%92%E7%AE%97%E6%B3%95%E7%9A%84%E6%97%B6%E9%97%B4%E5%A4%8D%E6%9D%82%E5%BA%A6%EF%BC%81.html

    public int fib2(int n){
        if(n<=1){
            return n;
        }
        return fib2(n-1)+fib2(n-2);
    }

70. 爬楼梯

题目链接

https://leetcode.cn/problems/climbing-stairs/description/

解题思路

多举几个例子,就可以发现其规律。
爬到第一层楼梯有一种方法,爬到二层楼梯有两种方法。
那么第一层楼梯再跨两步就到第三层 ,第二层楼梯再跨一步就到第三层。
所以到第三层楼梯的状态可以由第二层楼梯 和 到第一层楼梯状态推导出来,那么就可以想到动态规划了。

我们来分析一下,动规五部曲:

定义一个一维数组来记录不同楼层的状态
1.确定dp数组以及下标的含义
dp[i]: 爬到第i层楼梯,有dp[i]种方法

2.确定递推公式
如何可以推出dp[i]呢?
从dp[i]的定义可以看出,dp[i] 可以有两个方向推出来。
首先是dp[i - 1],上i-1层楼梯,有dp[i - 1]种方法,那么再一步跳一个台阶不就是dp[i]了么。
还有就是dp[i - 2],上i-2层楼梯,有dp[i - 2]种方法,那么再一步跳两个台阶不就是dp[i]了么。
那么dp[i]就是 dp[i - 1]与dp[i - 2]之和!
所以dp[i] = dp[i - 1] + dp[i - 2] 。
在推导dp[i]的时候,一定要时刻想着dp[i]的定义,否则容易跑偏。
这体现出确定dp数组以及下标的含义的重要性!

3.dp数组如何初始化
dp[1]=1 dp[2]=2,然后从i = 3开始递推,这样才符合dp[i]的定义。
4.确定遍历顺序
从递推公式dp[i] = dp[i - 1] + dp[i - 2];中可以看出,遍历顺序一定是从前向后遍历的
5.举例推导dp数组
举例当n为5的时候,dp table(dp数组)应该是这样的

如果代码出问题了,就把dp table 打印出来,看看究竟是不是和自己推导的一样。

code

class Solution {
    public int climbStairs(int n) {
        if(n<=1){
            return n;
        }
        //1.确定dp数组以及下标的含义
        //dp[i]: 爬到第i层楼梯,有dp[i]种方法
        //2.确定递推公式
        //dp[i]=dp[i-1]+dp[i-2]  dp[i-1]爬一个台阶到dp[i] dp[i-2]爬俩个台阶到dp[i] dp[3]=dp[2]+dp[1] 慢慢往上计算递推
        //3.dp数组如何初始化
        int[] dp=new int[n+1];
        dp[1]=1;
        dp[2]=2;
        //4.确定遍历顺序  后边有前边递推出结果,从前向后遍历
        for(int i=3;i<=n;i++){
            dp[i]=dp[i-1]+dp[i-2];
        }
        //5.举例推导dp数组
        //System.out.println(Arrays.toString(dp));
        return dp[n];
    }
}

优化空间(O)1

后面将讲解的很多动规的题目其实都是当前状态依赖前两个,或者前三个状态,都可以做空间上的优化,但我个人认为面试中能写出版本一就够了哈,清晰明了,如果面试官要求进一步优化空间的话,我们再去优化。

因为版本一才能体现出动规的思想精髓,递推的状态变化。

    int climbStairs(int n) {
        if (n <= 1) return n;
        int[] dp=new int[3];
        dp[1] = 1;
        dp[2] = 2;
        for (int i = 3; i <= n; i++) {
            int sum = dp[1] + dp[2];
            dp[1] = dp[2];
            dp[2] = sum;
        }
        return dp[2];
    }

746. 使用最小花费爬楼梯

题目链接

https://leetcode.cn/problems/min-cost-climbing-stairs/description/

解题思路

1.确定dp数组以及下标的含义
** dp[i]的定义:到达第i台阶所花费的最少体力为dp[i]。**
** 2.确定递推公式**
** 可以有两个途径得到dp[i],一个是dp[i-1] 一个是dp[i-2]。**
** dp[i - 1] 跳到 dp[i] 需要花费 dp[i - 1] + cost[i - 1]。**
** dp[i - 2] 跳到 dp[i] 需要花费 dp[i - 2] + cost[i - 2]。**
** 那么究竟是选从dp[i - 1]跳还是从dp[i - 2]跳呢?**
** 一定是选最小的,所以dp[i] = min(dp[i - 1] + cost[i - 1], dp[i - 2] + cost[i - 2]);**
3.如何初始化数组
只初始化dp[0]和dp[1] 其他均递推出来
新题目描述中明确说了 “你可以选择从下标为 0 或下标为 1 的台阶开始爬楼梯。”也就是说 到达 第 0 个台阶是不花费的,但从 第 0 个台阶 往上跳的话,需要花费 cost[0]。
所以初始化 dp[0] = 0,dp[1] = 0;
4.确定遍历顺序
** dp[i]由dp[i-1]dp[i-2]推出,所以是从前到后遍历cost数组就可以了**
** 5.举例推导dp数组 **
拿示例2:cost = [1, 100, 1, 1, 1, 100, 1, 1, 100, 1] ,来模拟一下dp数组的状态变化,如下

code

class Solution {
    public int minCostClimbingStairs(int[] cost) {
        //1.确定dp数组以及下标的含义
        //dp[i]的定义:到达第i台阶所花费的最少体力为dp[i]。
        //2.确定递推公式
        //可以有两个途径得到dp[i],一个是dp[i-1] 一个是dp[i-2]。
        // dp[i - 1] 跳到 dp[i] 需要花费 dp[i - 1] + cost[i - 1]。
        // dp[i - 2] 跳到 dp[i] 需要花费 dp[i - 2] + cost[i - 2]。
        // 那么究竟是选从dp[i - 1]跳还是从dp[i - 2]跳呢?
        // 一定是选最小的,所以dp[i] = min(dp[i - 1] + cost[i - 1], dp[i - 2] + cost[i - 2]);
        //3.如何初始化数组
        //只初始化dp[0]和dp[1] 其他均递推出来
        //新题目描述中明确说了 “你可以选择从下标为 0 或下标为 1 的台阶开始爬楼梯。”
        // 也就是说 到达 第 0 个台阶是不花费的,但从 第 0 个台阶 往上跳的话,需要花费 cost[0]。
        //所以初始化 dp[0] = 0,dp[1] = 0;
        int n=cost.length;
        int[] dp=new int[n+1];
        dp[0]=0;
        dp[1]=0;
        //4.确定遍历顺序
        //dp[i]由dp[i-1]dp[i-2]推出,所以是从前到后遍历cost数组就可以了
        for(int i=2;i<=n;i++){
            dp[i]=Math.min(dp[i-1]+cost[i-1],dp[i-2]+cost[i-2]);
        }
        //5.举例推导dp数组 ,拿示例2:cost = [1, 100, 1, 1, 1, 100, 1, 1, 100, 1] ,来模拟一下dp数组的状态变化,如下
        //System.out.println(Arrays.toString(dp));
        return dp[n];
    }
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

0208-场景状态模式的UML图

一、设置状态 二、Uml类图

白话Java - 守护线程

全文详见个人独立博客&#xff1a;白话Java - 守护线程 白话Java - 守护线程关于”白话”, 偶然想到的词。目的就是用简洁&#xff0c;明快的语言来告诉您&#xff0c;我所知道的一切。 Java中的线程分两类&#xff0c;用户线程和守护线程。 Thread commonThread new Thread(…

尚硅谷谷粒商城项目笔记——七、安装rabbitMQ【电脑CPU:AMD】

七、安装rabbitMQ 注意&#xff1a; 因为电脑是AMD芯片&#xff0c;自己知识储备不够&#xff0c;无法保证和课程中用到的环境一样&#xff0c;所以环境都是自己根据适应硬件软件环境重新配置的&#xff0c;这里的虚拟机使用的是VMware。 [!NOTE] 下载RabbitMQ和Erlang的安装…

最新!2024年—华为认证HCIP考证流程

HCIP HCIP&#xff08;Huawei Certified ICT Professional 华为认证ICT高级工程师&#xff09;是华为职业认证中用于标识个人能力在某一技术领域达到高级工程师级别的证明&#xff0c;表示通过认证的人员具有丰富的ICT知识和实践能力。 HCIP方向 HCIP认证条件 无&#xff0…

蝙蝠避障:盲人出行的守护者,让每一步都安心

在这个多彩的世界里&#xff0c;作为一位盲人&#xff0c;我时常感受着出行的种种挑战。每一次踏出家门&#xff0c;都是一场未知的探险&#xff0c;心里难免有些忐忑。街道上的车流声、人群的喧闹&#xff0c;这些对我而言只是模糊的背景音&#xff0c;真正让我担忧的是那些突…

【AI】OCR篇1

每日更新&#xff0c;建议关注、收藏、点赞 ocr流程 版面分析 、预处理-> 行列切割 -> 字符识别 -> 后处理识别矫正 判断页面上的文本朝向&#xff0c;图像预处理&#xff0c;做角度矫正和去噪。对文档版面进行分析&#xff0c;进每一行进行行分割&#xff0c;把每…

AI/机器学习(计算机视觉/NLP)方向面试复习4

1. 什么是知识图谱&#xff1f; 是结构化的语义知识库&#xff0c;用于迅速描述物理世界中的概念及其相互关系。它的基本单位是(实体-关系-实体) 就是点-线-点。 存储类型有结构化和非结构化的数据。结构化数据就是关系数据库&#xff0c;一张表格&#xff0c;关系很清楚&…

leetcode面试算法题

1.数组/字符串 1.1合并两个有序数组 class Solution {public void merge(int[] nums1, int m, int[] nums2, int n) {int p1 0;//遍历数组1int p2 0;//遍历数组2int[] nums3 new int[m n];int p 0;while (p1 < m && p2 < n) {//当数组1、2都没遍历完nums3[…

免费分享:中国1平方公里以上湖泊形状数据(附下载方法)

我国是世界上湖泊数量最多的国家之一&#xff0c;共有湖泊24800多个。其中面积在1平方千米以上的天然湖泊就有2800多个。湖泊分布呈现出显著的区域性差异。东部季风区&#xff0c;特别是长江中下游地区&#xff0c;分布着我国最大的淡水湖群&#xff1b;西部以青藏高原湖泊较为…

创新技术引领潮流,2024年录屏软件新技术

现在使用录屏工具的情况越来越多&#xff0c;大家可以用来录制游戏过程、可以用来录制教学视频、可以用来录制会议记录等需要记录屏幕操作的过程。那有什么适合我们自己的录屏工具吗&#xff1f;这次我给你介绍几款我用过的&#xff0c;你看看是不是符合你的需求吧。 1.福晰录…

多云应用安全平台RegData利用MongoDB简化数据控制和合规流程

在高度规范化市场中&#xff0c;为了保障数据安全&#xff0c;企业可能需要部署一系列繁琐且成本高昂的IT基础设施系统。随着各项数据安全保护措施的出台&#xff0c;企业需要遵守的法规数量越多&#xff0c;尤其是跨越多个地域的企业&#xff0c;其IT基础设施就会越复杂。如今…

《程序猿入职必会(9) · 用代码生成器快速开发》

&#x1f4e2; 大家好&#xff0c;我是 【战神刘玉栋】&#xff0c;有10多年的研发经验&#xff0c;致力于前后端技术栈的知识沉淀和传播。 &#x1f497; &#x1f33b; CSDN入驻不久&#xff0c;希望大家多多支持&#xff0c;后续会继续提升文章质量&#xff0c;绝不滥竽充数…

从文字到视频的魔法:CogVideoX让你的创意瞬间成真

CogVideoX 发布 如果有一个工具能够让你轻松跨越这些障碍&#xff0c;瞬间成为短视频高手&#xff0c;你会心动吗&#xff1f;今天&#xff0c;这个梦想终于成为现实。智谱AI最新推出的CogVideoX&#xff0c;正是为你量身打造的AI视频创作神器&#xff01; CogVideoX CogVideoX…

论文辅导 | 基于时空Transformer 网络的隧道交通运行风险 动态辨识方法

辅导文章 模型描述 为了及时发现、评估与应对高速公路隧道交通风险隐患&#xff0c;确保隧道运行安全通畅&#xff0c;构建了基于时空Transformer网络的隧道运行风险状态动态辨识方法。以隧道交通流全域检测数据与关键断面集计数据为输入&#xff0c;通过空间CNN卷积与时序LST…

蜂窝物联智慧农业新篇章:揭秘智慧大棚的科技魅力!

在这个科技日新月异的时代&#xff0c;农业这一古老而重要的行业正经历着前所未有的变革。智慧农业的兴起&#xff0c;尤其是智慧大棚的应用&#xff0c;正悄然改变着我们对传统农业的认知&#xff0c;引领着农业生产进入了一个高效、精准、可持续发展的新时代。今天&#xff0…

怎么查询大数据信用评分?

相信在了解大数据信用评分的时候&#xff0c;不少人都因为大数据信用评分在申贷的时候遭受到过挫折&#xff0c;因为大数据信用已经被很多银行和金融机构作为风险控制的重要依据使用&#xff0c;其中的大数据信用评分&#xff0c;能直观的感知到用户的信用情况。那如何查询大数…

【ESP-IDF】ESP32获取真随机数

ESP32内置一个真随机数发生器。我翻了好几个型号的ESP32的编程指南&#xff0c;都有相关的函数&#xff0c;应该是大部分型号都有&#xff0c;具体有没有还需要对照一下自己手上的ESP32型号。 我们可以从从随机数发生器的寄存器 RNG_DATA_REG 中读取随机数&#xff0c;每个读到…

到底哪款蓝牙耳机才是性价比之王?精选四款平价高品质耳机品牌!

在当前的音频设备市场中&#xff0c;蓝牙耳机已成为众多现代人挑选的热门配件。选蓝牙耳机不同人看重的点不一样&#xff0c;但倘若像我这样喜欢用来听音乐的&#xff0c;就千万别忽视耳机的音质。现在的产品越来越多&#xff0c;到底哪款蓝牙耳机才是性价比之王&#xff1f;一…

html+css 实现hover背景彩色按钮

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享htmlcss 绚丽效果&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 文…

数据治理数据管理体系:数据标准、数据中台、数据安全、大数据平台与架构

数据驱动&#xff0c;资产为王&#xff0c;企业竞争的核心逐渐从传统的资源争夺转向了数据价值的挖掘与利用。数据&#xff0c;作为企业的新石油&#xff0c;正以前所未有的速度重塑着商业格局。为了在这场数据革命中占据先机&#xff0c;构建一套完善的数据治理体系显得尤为重…