代码随想录算法训练营DAY38|C++动态规划Part.1|动态规划理论基础、509.斐波那契数、70.爬楼梯、746.使用最小花费爬楼梯

文章目录

  • 动态规划理论基础
    • 什么是动态规划
    • 动态规划的解题步骤
      • DP数组以及下标的含义
      • 递推公式
      • DP数组初始化
      • DP数组遍历顺序
      • 打印DP数组
      • 动态规划五部曲
    • 动态规划应该如何debug
  • 509.斐波那契数
    • 什么是斐波那契数列
    • 动态规划五部曲
      • 确定dp数组下标以及含义
      • 确定递推公式
      • dp数组如何初始化
      • 确定遍历顺序
      • 打印DP数组
    • 代码实现
    • CPP代码
  • 70.爬楼梯
    • 题意分析
    • 动规五部曲
      • 确定dp数组下标以及含义
      • 确定递推公式
      • dp数组如何初始化
      • 确定遍历顺序
      • 打印DP数组
    • 扩展题
    • CPP代码
  • 746.使用最小花费爬楼梯

在这里插入图片描述

动态规划理论基础

什么是动态规划

动态规划(Dynamic Programming, DP),如果某一个问题有很多重叠子问题,这样往往是用动态规划是最有效的。

所以动态规划中每一个状态一定是由上一个状态推导出来的这一点就区分于贪心,贪心没有状态推导,而是从局部直接选最优的。

对于刷题来说就抓住一句话:

动规是由前一个状态推导出来的,而贪心是从局部直接选出最优的

动态规划的解题步骤

在全面总结动态规划解题步骤前,同时也是做动态规划类题目之前,我们要先搞明白以下几个问题:

DP数组以及下标的含义

有很多类问题有一个二维dp数组,其中的行、列的含义一定要搞清楚;同理一维dp数组中的值又是什么呢?在写代码前,一定要搞明白这是什么意思。

递推公式

动态规划递推公式仅仅是一部分,而不是只要掌握了递推公式,动规就变得简单了。

DP数组初始化

dp数组如何初始化其实紧贴着dp数组以及下标含义,如果dp数组里是什么,各个维度含义是什么,那dp数组的初始化更加无从谈起了。

DP数组遍历顺序

对于背包类问题,遍历顺序是非常有讲究的,所以每道题一定要弄清他们的遍历顺序。

打印DP数组

如果题目通过不了,首先应该考虑是不是dp数组出了问题,我们应该吧dp数组打印出来看是否符合预期

动态规划五部曲

一定要仔细分析上述的五大问题

  1. 确定dp数组(dp table)以及下标的含义
  2. 确定递推公式
  3. dp数组如何初始化
  4. 确定遍历顺序
  5. 举例推导dp数组

动态规划应该如何debug

做动规的题目,写代码之前一定要把状态转移在dp数组的上具体情况模拟一遍,心中有数,确定最后推出的是想要的结果

然后再写代码,如果代码没通过就打印dp数组,看看是不是和自己预先推导的哪里不一样。

这样才是一个完整的思考过程,而不是一旦代码出问题,就毫无头绪的东改改西改改,最后过不了,或者说是稀里糊涂的过了

这里给出卡哥写的自我debug的灵魂三问:

  • 这道题目我举例推导状态转移公式了么?
  • 我打印dp数组的日志了么?
  • 打印出来了dp数组和我想的一样么?

509.斐波那契数

力扣题目链

文章链接:509.斐波那契数列

视频链接:手把手带你入门动态规划 | leetcode:509.斐波那契数

状态:迭代法还是会写一写

什么是斐波那契数列

动态规划五部曲

确定dp数组下标以及含义

dp[i]表示第i个斐波那契数值为dp[i]

确定递推公式

斐波那契数的递推公式很明显: d p [ i ] = d p [ i − 1 ] + d p [ i − 2 ] dp[i] = dp[i - 1] + dp[i - 2] dp[i]=dp[i1]+dp[i2]

dp数组如何初始化

斐波那契的初始化也很简单 [ 0 , 1 , 1 , 2 , 3 , 5 , 8 ] [0, 1, 1, 2, 3, 5, 8] [0,1,1,2,3,5,8]

由数学归纳可知: d p [ 0 ] = 0 ; d p [ 1 ] = 1 ; dp[0] = 0; dp[1] = 1; dp[0]=0;dp[1]=1;

在很多题目中,dp数组的初始化往往也依赖于递推公式

确定遍历顺序

本题中,遍历顺序从前向后即可。(有一些题目从后向前遍历;有一些题目两层for循环,其先先后顺序也有说法)

打印DP数组

这个主要是用来debug的

代码实现

vector<int> dp(n+1);
//初始化
dp[0] = 0; dp[1] = 1;
for (i = 2; i <= n; i++)
  dp[i] = dp[i-1] + dp[i-2];
return dp[n];

对状态进行压缩,只需要维护两个数值就可以了,不需要记录整个序列.代码可以缩减为

int fib(int N){
  	if (N <= 1) return N;
  	int dp[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)的递归解法:

int fib(int N){
  if (N < 2) return N;
  return fib(N - 1) + fib(N - 2);
}

CPP代码

class Solution {
public:
    int fib(int N) {
        if (N <= 1) return N;
        vector<int> dp(N + 1);
        dp[0] = 0;
        dp[1] = 1;
        for (int i = 2; i <= N; i++) {
            dp[i] = dp[i - 1] + dp[i - 2];
        }
        return dp[N];
    }
};
//只维护两个数值
class Solution {
public:
    int fib(int N) {
        if (N <= 1) return N;
        int dp[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];
    }
};
//递归
class Solution {
public:
    int fib(int N) {
        if (N < 2) return N;
        return fib(N - 1) + fib(N - 2);
    }
};

70.爬楼梯

力扣题目链接

文章讲解:70.爬楼梯

视频讲解:带你学透动态规划-爬楼梯|LeetCode:70.爬楼梯)

状态:

题意分析

如果对于一个n阶台阶,每次你可以爬 12 个台阶。有多少种不同的方法可以爬到楼顶。

如果台阶一共只有1阶,一共只有1

如果台阶有2阶,一共有2

如果台阶有3阶,一共有3

如果台阶有4阶,一共有5

本质上,当前台阶有几种方法,主要依赖于前两个状态

其实这就是递推关系。这种递推关系可以用动态规划来解决。

动规五部曲

确定dp数组下标以及含义

dp[i],达到i阶有dp[i]种方法。

确定递推公式

dp[i - 2]表示达到i-2阶有dp[i - 2]种方法

dp[i - 1]表示达到i-1阶有dp[i - 1]种方法

dp[i]表示达到i阶有dp[i]种方法

d p [ i ] = d p [ i − 2 ] + d p [ i − 1 ] dp[i] = dp[i-2]+dp[i-1] dp[i]=dp[i2]+dp[i1]

dp数组如何初始化

从递推公式可以看出,最重要的就是初始化数组的前几位

dp[0] = 1还是dp[0] = 0

因为我们dp[2] = 2,从递推公式出发,dp[0]应该初始化成1。同时题目中也说过,n为正整数

在代码随想录的代码中,是不初始化0的,而是直接初始化dp[2]dp[1]

确定遍历顺序

从前向后遍历即可

打印DP数组

用于debug

其实本质上,从上文论述的规律可以看出,其实本题就是一个斐波那契数列

扩展题

一步可以走m个台阶,爬n阶的楼梯有多少种方法。

词题可以用完全背包的思路来解决。

CPP代码

//还是只维护两个数组
class Solution {
public:
    int climbStairs(int n) {
        if (n <= 3) return n;
        int dp[2];
        dp[0] = 1; dp[1] = 2;
        for (int i = 3; i <= n; ++i) {
            int cur = dp[0] + dp[1];
            dp[0] = dp[1];
            dp[1] = cur;
        }
        return dp[1];

    }
};

746.使用最小花费爬楼梯

力扣题目链接

文章讲解:746.使用最小花费爬楼梯

视频讲解:动态规划开更了!| LeetCode:746. 使用最小花费爬楼梯

状态:相较于前两个题目,难度一下就上来了,我个人感觉是贪心和动态规划的结合。

在题目描述中,我们可以选择从下标为0或下标为1的台阶开始爬楼梯。

思路

dp数组含义

想给出结论,本题仍然只需要一个一维的dp数组,其下标记录我们到达了哪个台阶,其中的值就是体力的最小消耗。

以上的推导都是怎么来的呢?

要求的是爬到楼顶,那么我们dp数组应该得用下标来表示我们爬的位置,看是不是到楼顶了;

然后我们要求一个最小消耗,那么我们的dp数组刚好可以存到达该层的一个消耗,刚好逻辑就闭环。

综上:

dp[i]表示到达i位置所需要的花费为dp[i]

递推公式

本题中我们要求的就是dp[i],那么如何跳到dp[i]呢?我们可以从dp[i-1]跳一步到dp[i],也可以从dp[i-2]跳两步到dp[i]

如果从dp[i-1]跳到dp[i]的话,所需要的花费是dp[i-1] + cost[i-1]

如果从dp[i-2]跳到dp[i]的话,所需要的花费是dp[i-2] + cost[i-2]

dp[i]表示到达i位置所需要的花费为dp[i]

显然递推公式为: d p [ i ] = m i n ( d p [ i − 1 ] + c o n s t [ i − 1 ] , d p [ i − 2 ] + c o n s t [ i − 2 ] ) dp[i] = min(dp[i-1] + const[i-1], dp[i-2] + const[i-2]) dp[i]=min(dp[i1]+const[i1],dp[i2]+const[i2])

初始化

很明显,本题只要初始化了dp[1]dp[0]就可以把递推公式打通了。

//题意表明了,可以选择从0层或者从第1层开始跳
dp[0] = 0;	//到达0位置最小花费为0
dp[1] = 0;	//到底1位置最小花费也是0. 因为我们可以选择嘛

遍历顺序

从零往后去遍历

因为是模拟台阶,而且dp[i]由dp[i-1]dp[i-2]推出,所以是从前到后遍历cost数组就可以了。

打印dp数组

打印出来看是不是和我们预想的一样

CPP代码

class Solution {
public:
    int minCostClimbingStairs(vector<int>& cost) {
        vector<int> dp(cost.size() + 1);
        dp[0] = 0; // 默认第一步都是不花费体力的
        dp[1] = 0;
        for (int i = 2; i <= cost.size(); i++) {
            dp[i] = min(dp[i - 1] + cost[i - 1], dp[i - 2] + cost[i - 2]);
        }
        return dp[cost.size()];
    }
};

//只维护两个数字
class Solution {
public:
    int minCostClimbingStairs(vector<int>& cost) {
        int dp0 = 0;
        int dp1 = 0;
        for (int i = 2; i <= cost.size(); i++) {
            int dpi = min(dp1 + cost[i - 1], dp0 + cost[i - 2]);
            dp0 = dp1; // 记录一下前两位
            dp1 = dpi;
        }
        return dp1;
    }
};

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

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

相关文章

场景文本检测识别学习 day07(BERT论文精读)

BERT 在CV领域&#xff0c;可以通过训练一个大的CNN模型作为预训练模型&#xff0c;来帮助其他任务提高各自模型的性能&#xff0c;但是在NLP领域&#xff0c;没有这样的模型&#xff0c;而BERT的提出&#xff0c;解决了这个问题BERT和GPT、ELMO的区别&#xff1a; BERT是用来…

翻译《The Old New Thing》 - Why .shared sections are a security hole

Why .shared sections are a security hole - The Old New Thing (microsoft.com)https://devblogs.microsoft.com/oldnewthing/20040804-00/?p38253 Raymond Chen 2004年08月04日 许多人会推荐使用共享数据节作为在应用程序的多个实例之间共享数据的一种方式。这听起来是个好…

走向大规模应用之前,DePIN 如何突破技术、数据与市场之网

近期&#xff0c;随着分布式物理基础设施网络&#xff08;DePIN&#xff09;的快速演变&#xff0c;一个旨在利用区块链技术彻底改造传统基础设施模型的新兴生态系统正在逐渐浮现。2024 年 4 月&#xff0c;以 peaq 为代表的 DePIN 项目成功筹集了 1500 万美元用于生态系统的扩…

通过 API从 0 到 1 构建 本地 GPTs——1.构建Builder‘s Prompt

目的&#xff1a;帮助小白用户生成结构化 prompt 功能清单 搭建本地 gpts 能力&#xff0c;构建本地企业知识库助手Builder’s Prompt -对话引导构建 prompt 示例&#xff0c;生成助手信息function_call的用法prompt 示例 GPTs 的 Create 能力 用于引导用户构建结构化的 pr…

深度学习的瓶颈是什么!

深度学习主要的瓶颈&#xff1a; 数据依赖与标注问题&#xff1a;深度学习模型通常需要大量的标注数据来进行训练。然而&#xff0c;获取大量的标注数据不仅成本高昂&#xff0c;而且在某些领域&#xff08;如医疗、金融等&#xff09;中可能难以获取足够的标注数据。此外&…

python-excel自动化-openpyxl

openpyxl学习笔记 创建或打开表格存储和遍历数据设置单元格风格过滤器和排序更改工作表的背景颜色合并单元格冻结窗口数字格式公式图像图表条形图折线图散点图 创建或打开表格 # 创建 import datetime from openpyxl import Workbook # 实例化 wb Workbook() # 激活 work…

四:物联网ARM开发

一&#xff1a;ARM体系结构概述 1&#xff1a;控制外设led灯还有一些按键这些就要用到gpio&#xff0c;采集传感器的数据需要adc进行转化数据格式&#xff0c;特殊的外设和传感器是通过特殊的协议接口去进行连接的比如一些轴传感器和主控器的连接是通过spi&#xff0c;IIC 控制…

Check the `candidate.safety_ratings` to see if the respoe was blocked.

ValueError&#xff1a;“response.text”快速访问器仅适用于简单&#xff08;单“部分”&#xff09;文本响应。此响应不是简单的文本。请改用“result.parts”访问器或完整的“result.candidates[index].content.parts”查找。期号 #170 谷歌-双子座/生成-人工智能-python Gi…

JavaScript 日期对象

在 JavaScript 中&#xff0c;你可以使用 Date 对象来处理日期和时间。以下是一些常见的 Date 对象的使用方法&#xff1a; 1、创建日期对象&#xff1a; // 创建一个表示当前日期和时间的 Date 对象 let currentDate new Date();// 创建一个特定日期和时间的 Date 对象 let…

GPB | RegVar:基于深度神经网络的非编码区突变功能预测新方法

Genomics, Proteomics & Bioinformatics &#xff08;GPB&#xff09;发表了由军事医学研究院辐射医学研究所张成岗研究员、周钢桥研究员和卢一鸣副研究员团队完成的题为“RegVar: Tissue-specific Prioritization of Noncoding Regulatory Variants”的方法文章。我们的“…

数据结构 - 栈

目录 一. 栈的概念 二. 栈的结构 三. 栈的实现 1. 实现栈的两种方式 链表实现栈 顺序表实现栈 选择依据 栈的创建 栈的初始化 栈的销毁 入栈 出栈 获取栈顶元素 判断栈是否为空 获取栈中有效数据的个数 一. 栈的概念 栈&#xff08;Stack&#xff09;是一种重要…

VScode Failed to parse remote port from server output

在使用VScode 在连接AutoDL 过程中一直连接不上&#xff0c;显示 Failed to parse remote port from server output 在网上查了很多资料&#xff0c;貌似的没啥用。和我有相同 error 的可以尝试修改setting.json 文件。 添加这条命令&#xff08;我的json文件里面没有&#…

共享购:融合社交分享与消费返利的创新电商模式

共享购电商模式是一种独特的商业模式&#xff0c;巧妙地将社交分享与消费返利结合&#xff0c;让消费者在购物的同时&#xff0c;也能通过平台资产奖励实现价值的双重增长。该平台资产体系主要由共享值和共享积分两大要素构成&#xff0c;共同构建了一个充满活力的电商生态系统…

区块链技术与应用学习笔记(8-9节)——北大肖臻课程

目录 8.挖矿 对于全节点和轻节点思考问题&#xff1f; ①全节点在比特币的主要作用&#xff1f; ②挖矿时当监听到别人已经挖出区块并且延申了最长合法链此时应该立刻放弃当前区块在 本地重新组装一个指向最后这个新合法区块的候选区块&#xff0c;重新开始挖矿。节点这么做…

vivado 使用“链路 (Links)”窗口查看和更改链路设置

使用“链路 (Links) ”窗口查看和更改链路设置 创建链路后 &#xff0c; 就会将其添加到“ Links ”视图 &#xff08; 请参阅下图 &#xff09; 中 &#xff0c; 该视图是更改链路设置和查看状态的主要方法 &#xff0c; 也是最佳方法。 “ Links ”窗口中的每一行都对应 1 …

pymilvus创建多向量

pymilvus创建多向量 从 Milvus 2.4 开始&#xff0c;引入了多向量支持和混合搜索框架&#xff0c;单个collection可以支持10个向量字段。不同的向量字段可以表示不同的方面、不同的embedding模型甚至表征同一实体的不同数据模态。该功能在综合搜索场景中特别有用&#xff0c;例…

python学习笔记----python基础语法(二)

一、字面量 在 Python 中&#xff0c;字面量 是一种直接在代码中表示其自身值的数据。字面量用于创建值&#xff0c;并且可以直接被 Python 的解释器识别和处理。不同类型的数据有不同的字面量形式。下面是一些常见的字面量类型&#xff1a; 二、注释 注释&#xff1a;在程序…

[Android14] SystemUI的启动

1. 什么是System UI SystemUI是Android系统级应用&#xff0c;负责反馈系统及应用状态并与用户保持大量的交互。业务主要涉及的组成部分包括状态栏(Status Bar)&#xff0c;通知栏(Notification Panel)&#xff0c;锁屏(Keyguard)&#xff0c;控制中心(Quick Setting)&#xff…

Babylon.js和Three.js的区别

Babylon.js和Three.js都是基于WebGL的3D图形库&#xff0c;它们使得开发者能够在网页上创建和展示3D内容。尽管它们的目标相似&#xff0c;但在设计理念、功能集、性能和社区支持等方面存在一些差异。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢…

SpringCloud引入SpringBoot Admin

Spring Boot Admin可以监控和管理Spring Boot&#xff0c;能够将 Actuator 中的信息进行界面化的展示&#xff0c;也可以监控所有 Spring Boot 应用的健康状况&#xff0c;提供警报功能。 1. 创建SpringBoot工程 2. 引入相关依赖 <dependency><groupId>com.alib…
最新文章