算法沉淀——动态规划之斐波那契数列模型(leetcode真题剖析)

news2024/12/25 0:30:40

在这里插入图片描述

算法沉淀——动态规划之斐波那契数列模型

  • 01.第 N 个泰波那契数
  • 02.三步问题
  • 03.使用最小花费爬楼梯
  • 04.解码方法

动态规划(Dynamic Programming,简称DP)是一种通过将原问题分解为相互重叠的子问题并仅仅解决每个子问题一次,将其解存储起来,避免重复计算,从而提高效率的算法优化技术。它通常用于求解最优化问题。

动态规划的基本思想是利用之前已经计算过的结果,通过递推关系式来计算当前问题的解。它具有以下关键特点:

  1. 最优子结构: 问题的最优解可以通过其子问题的最优解来构造。
  2. 重叠子问题: 原问题可以分解为若干个子问题,而这些子问题可能会被多次重复求解。
  3. 状态转移方程: 动态规划通过定义状态和状态之间的转移关系来描述问题。通过状态转移方程,我们可以推导出问题的最优解。

动态规划可以分为两种主要类型:自顶向下的记忆化搜索和自底向上的递推法。

  • 自顶向下(Top-Down): 利用递归的方式,通过递归调用来解决问题,并通过记忆化技术来避免重复计算,即缓存已经计算过的子问题的结果。
  • 自底向上(Bottom-Up): 从最小的子问题开始,通过迭代的方式逐步求解规模更大的问题,通常使用数组或表格来存储子问题的结果,避免了递归的开销。

动态规划常用于求解具有最优子结构和重叠子问题性质的问题,例如最短路径问题、背包问题、编辑距离问题等。

典型的动态规划问题包括:

  • 斐波那契数列: F(n) = F(n-1) + F(n-2)。
  • 最长上升子序列(LIS): 求一个序列中最长的递增子序列的长度。
  • 背包问题: 在给定限制条件下,选择最优的物品放入背包,使得价值最大。
  • 最短路径问题: 求图中两个节点之间的最短路径。
  • 编辑距离问题: 计算两个字符串之间的最小编辑操作次数。

动态规划算法是一种强大的算法设计思想,可以在很多问题中得到应用,但需要注意问题是否满足最优子结构和重叠子问题的条件。

01.第 N 个泰波那契数

题目链接:https://leetcode.cn/problems/n-th-tribonacci-number/

泰波那契序列 Tn 定义如下:

T0 = 0, T1 = 1, T2 = 1, 且在 n >= 0 的条件下 Tn+3 = Tn + Tn+1 + Tn+2

给你整数 n,请返回第 n 个泰波那契数 Tn 的值。

示例 1:

输入:n = 4
输出:4
解释:
T_3 = 0 + 1 + 1 = 2
T_4 = 1 + 1 + 2 = 4

示例 2:

输入:n = 25
输出:1389537

提示:

  • 0 <= n <= 37
  • 答案保证是一个 32 位整数,即 answer <= 2^31 - 1

思路

动态规划的题目主要分为5个步骤。

1.状态表示:这道题可以根据题目要求直接给出,dp[i]等于第i个泰波纳契数。

2.状态转移方程:同样原题已经给出,dp[i] = dp[i - 1] + dp[i - 2] + dp[i - 3] 。

3.初始化:我们只需要将前三个数初始化即可,防止越界。

4.填表顺序:从左往右。

5.返回值:dp[n]。

代码

class Solution {
public:
    int tribonacci(int n) {
        int dp[38];
        dp[0]=0,dp[1]=dp[2]=1;
        for(int i=3;i<=n;++i)
            dp[i]=dp[i-1]+dp[i-2]+dp[i-3];
        
        return dp[n];
    }
};

02.三步问题

题目链接:https://leetcode.cn/problems/three-steps-problem-lcci/

三步问题。有个小孩正在上楼梯,楼梯有n阶台阶,小孩一次可以上1阶、2阶或3阶。实现一种方法,计算小孩有多少种上楼梯的方式。结果可能很大,你需要对结果模1000000007。

示例1:

 输入:n = 3 
 输出:4
 说明: 有四种走法

示例2:

 输入:n = 5
 输出:13

提示:

  1. n范围在[1, 1000000]之间

思路

  1. 状态表示: 定义状态 dp[i] 表示到达第 i 个位置时的总方法数。
  2. 状态转移方程: 根据题目要求,dp[i] 可以由前三个状态转移而来:dp[i - 1]dp[i - 2]dp[i - 3]。因此,状态转移方程为 dp[i] = dp[i - 1] + dp[i - 2] + dp[i - 3]
  3. 取模操作: 由于题目要求结果取模,需要在每次计算时都进行取模操作,防止整数溢出。
  4. 初始化: 需要初始化前三个位置的值,即 dp[1] = 1dp[2] = 2dp[3] = 4
  5. 填表顺序: 从左到右填表,逐步计算出 dp[i] 的值。
  6. 返回值: 返回 dp[n] 的值,即到达第 n 个位置时的总方法数。

这样的动态规划问题通常可以通过填表的方式解决,遵循以上步骤可以得到最终的解。

代码

class Solution {
    const int Mod = 1e9+7;
public:
    int waysToStep(int n) {
        if(n==1||n==2) return n;
        if(n==3) return 4;
        vector<long long> dp(n+1);
        dp[1]=1,dp[2]=2,dp[3]=4;
        for(int i=4;i<=n;i++)
            dp[i]=(dp[i-1]+dp[i-2]%Mod+dp[i-3])%Mod;
        return dp[n];
    }
};

03.使用最小花费爬楼梯

题目链接:https://leetcode.cn/problems/min-cost-climbing-stairs/

给你一个整数数组 cost ,其中 cost[i] 是从楼梯第 i 个台阶向上爬需要支付的费用。一旦你支付此费用,即可选择向上爬一个或者两个台阶。

你可以选择从下标为 0 或下标为 1 的台阶开始爬楼梯。

请你计算并返回达到楼梯顶部的最低花费。

示例 1:

输入:cost = [10,15,20]
输出:15
解释:你将从下标为 1 的台阶开始。
- 支付 15 ,向上爬两个台阶,到达楼梯顶部。
总花费为 15 。

示例 2:

输入:cost = [1,100,1,1,1,100,1,1,100,1]
输出:6
解释:你将从下标为 0 的台阶开始。
- 支付 1 ,向上爬两个台阶,到达下标为 2 的台阶。
- 支付 1 ,向上爬两个台阶,到达下标为 4 的台阶。
- 支付 1 ,向上爬两个台阶,到达下标为 6 的台阶。
- 支付 1 ,向上爬一个台阶,到达下标为 7 的台阶。
- 支付 1 ,向上爬两个台阶,到达下标为 9 的台阶。
- 支付 1 ,向上爬一个台阶,到达楼梯顶部。
总花费为 6 。

提示:

  • 2 <= cost.length <= 1000
  • 0 <= cost[i] <= 999

思路一

  1. 状态表示: 定义状态 dp[i] 表示到达第 i 个位置时的最小花费。注意:到达第 i 个位置的时候,第 i 位置的花费不需要算上。
  2. 状态转移方程: 根据题目要求,dp[i] 可以由前两个状态转移而来: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,因为不需要任何花费就可以直接站在第 0 层和第 1 层上。
  4. 填表顺序: 从左到右填表,逐步计算出 dp[i] 的值。
  5. 返回值: 返回 dp[n] 的值,即到达第 n 个位置时的最小花费。

代码一

class Solution {
public:
    int minCostClimbingStairs(vector<int>& cost) {
        int n=cost.size();
        vector<int> dp(n+1,0);
        dp[0]=dp[1]=0;
        for(int i=2;i<=n;++i)
            dp[i]=min(cost[i-1]+dp[i-1],cost[i-2]+dp[i-2]);
        return dp[n];
    }
};

思路二

  1. 状态表示: 定义状态 dp[i] 表示从第 i 个位置出发到达楼顶时的最小花费。
  2. 状态转移方程: 根据题目要求,dp[i] 可以由后两个状态转移而来:dp[i + 1]dp[i + 2]。因此,状态转移方程为 dp[i] = min(dp[i + 1], dp[i + 2]) + cost[i]
  3. 初始化: 为了保证填表的时候不越界,需要初始化最后两个位置的值,结合状态表表示得到 dp[n - 1] = cost[n - 1]dp[n - 2] = cost[n - 2]
  4. 填表顺序: 从右到左填表,逐步计算出 dp[i] 的值。
  5. 返回值: 返回 dp[0]dp[1]中最小的值。

代码二

class Solution {
public:
    int minCostClimbingStairs(vector<int>& cost) {
        int n=cost.size();
        vector<int> dp(n,0);
        dp[n-1]=cost[n-1],dp[n-2]=cost[n-2];
        for(int i=n-3;i>=0;--i)
            dp[i]=min(cost[i]+dp[i+1],cost[i]+dp[i+2]);
        return min(dp[0],dp[1]);
    }
};

04.解码方法

题目链接:https://leetcode.cn/problems/decode-ways/

一条包含字母 A-Z 的消息通过以下映射进行了 编码

'A' -> "1"
'B' -> "2"
...
'Z' -> "26"

解码 已编码的消息,所有数字必须基于上述映射的方法,反向映射回字母(可能有多种方法)。例如,"11106" 可以映射为:

  • "AAJF" ,将消息分组为 (1 1 10 6)
  • "KJF" ,将消息分组为 (11 10 6)

注意,消息不能分组为 (1 11 06) ,因为 "06" 不能映射为 "F" ,这是由于 "6""06" 在映射中并不等价。

给你一个只含数字的 非空 字符串 s ,请计算并返回 解码 方法的 总数

题目数据保证答案肯定是一个 32 位 的整数。

示例 1:

输入:s = "12"
输出:2
解释:它可以解码为 "AB"(1 2)或者 "L"(12)。

示例 2:

输入:s = "226"
输出:3
解释:它可以解码为 "BZ" (2 26), "VF" (22 6), 或者 "BBF" (2 2 6) 。

示例 3:

输入:s = "06"
输出:0
解释:"06" 无法映射到 "F" ,因为存在前导零("6" 和 "06" 并不等价)。

提示:

  • 1 <= s.length <= 100
  • s 只包含数字,并且可能包含前导零。

思路

  1. 状态表示: 定义状态 dp[i] 表示字符串中 [0, i] 区间上,一共有多少种编码方法。

  2. 状态转移方程: 分析 i 位置的 dp 值,有两种解码情况:

    • s[i] 在区间 [1, 9] 内时,说明 s[i] 可以单独解码,此时 dp[i] += dp[i - 1]
    • s[i - 1]s[i] 结合后在区间 [10, 26] 内时,说明前两个字符有一种编码方式,此时 dp[i] += dp[i - 2]

    综上所述,状态转移方程为:

    dp[i] = (s[i] != '0' ? dp[i - 1] : 0) + (10 <= stoi(s.substr(i - 1, 2)) <= 26 ? dp[i - 2] : 0)
    
  3. 初始化: 需要初始化前两个位置的值:

    • 初始化 dp[0]
      • s[0] == '0' 时,没有编码方法,结果 dp[0] = 0
      • s[0] != '0' 时,能编码成功,此时 dp[0] = 1
    • 初始化 dp[1]
      • s[1] 在区间 [1, 9] 内时,能单独编码,此时 dp[1] += dp[0]
      • s[0]s[1] 结合后的数在区间 [10, 26] 内时,说明前两个字符有一种编码方式,此时 dp[1] += 1
  4. 填表顺序: 从左往右填表,逐步计算出 dp[i] 的值。

  5. 返回值: 返回 dp[n - 1] 的值,表示在 [0, n - 1] 区间上的编码方法数量。

代码

class Solution {
public:
    int numDecodings(string s) {
        int n=s.size();
        vector<int> dp(n);

        dp[0]=s[0]!='0';
        if(n==1) return dp[0];

        if(s[1]>='1'&&s[1]<='9') dp[1]+=dp[0];
        int t=(s[0]-'0')*10+s[1]-'0';
        if(t>=10&&t<=26) dp[1]+=1;

        for(int i=2;i<n;++i){
            if(s[i]>='1'&&s[i]<='9') dp[i]+=dp[i-1];
            int t=(s[i-1]-'0')*10+s[i]-'0';
            if(t>=10&&t<=26) dp[i]+=dp[i-2];
        }

        return dp[n-1];
    }
};

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

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

相关文章

Linux日志轮替

文章目录 1. 基本介绍2. 日志轮替文件命名3. logrotate 配置文件4. 把自己的日志加入日志轮替5. 日志轮替机制原理6. 查看内存日志 1. 基本介绍 日志轮替就是把旧的日志文件移动并改名&#xff0c;同时建立新的空日志文件&#xff0c;当旧日志文件超出保存的范围之后&#xff…

深度学习基础(四)医疗影像分析实战

之前的章节我们初步介绍了卷积神经网络&#xff08;CNN&#xff09;和循环神经网络&#xff08;RNN&#xff09;&#xff1a; 深度学习基础&#xff08;三&#xff09;循环神经网络&#xff08;RNN&#xff09;-CSDN博客文章浏览阅读1.2k次&#xff0c;点赞17次&#xff0c;收…

window: C++ 获取自己写的dll的地址

我自己用C写了一个插件,插件是dll形式的,我的插件式在dll的目录下有个config文件夹,里面是我用json写的插件配置文件,当插件运行的时候我需要读取到json配置文件,所有最重要的就是如何获取dll的路径. 大概就是这么个结构, 我自己封装了一个函数.只适用于window编程,因为里面用…

个人博客系列-前端部署-创建框架(4)

项目环境介绍 Vue3 Vite TypeScript 服务器&#xff1a;阿里云contos node版本&#xff1a;v18.18.2 npm版本&#xff1a;v10.2.4 执行下面一行命令&#xff0c;创建vue3框架 npm create vuelatest修改端口&#xff1a;9528&#xff0c; 此步骤可以忽略&#xff08;使用默…

十三、集合进阶——双列集合

集合进阶——双列集合 13.1 双列集合的特点13.2 Map集合13.2.1 Map集合常用的API13.2.2 Map的遍历方式 13.3 HashMap13.4 LinkedHashMap13.5 TreeMap13.6 源码解析HashMap源码解读TreeMap源码解读 13.7 可变参数13.8 Collections13.9综合练习 13.1 双列集合的特点 双列集合一次…

【动态规划专栏】动态规划:似包非包---不同的二叉树

本专栏内容为&#xff1a;算法学习专栏&#xff0c;分为优选算法专栏&#xff0c;贪心算法专栏&#xff0c;动态规划专栏以及递归&#xff0c;搜索与回溯算法专栏四部分。 通过本专栏的深入学习&#xff0c;你可以了解并掌握算法。 &#x1f493;博主csdn个人主页&#xff1a;小…

SmartX 携手 openGauss 社区发布联合方案评测与性能最佳实践

近日&#xff0c;北京志凌海纳科技有限公司&#xff08;以下简称 “SmartX”&#xff09;携手 openGauss 社区完成了 openGauss 数据库基于 SmartX 超融合平台&#xff08;SMTX OS&#xff09;和 SmartX 分布式存储平台&#xff08;SMTX ZBS&#xff09;的性能测试和调优。 结果…

【C++】模板初阶 | 泛型编程 | 函数模板 | 类模板

目录 1. 泛型编程 2. 函数模板 2.1 函数模板概念 2.2 函数模板格式 2.3 函数模板的原理 2.4 函数模板的实例化 2.5 模板参数的匹配原则 3. 类模板 3.1 类模板的定义格式 3.2 类模板的实例化 【本节目标】 1. 泛型编程 2. 函数模板 3. 类模板 1. 泛型编程 如何实现一…

C语言调试

目录 一.Debug和Release介绍 二.Windows环境调试介绍 三.窗口介绍 &#xff08;1&#xff09;自动窗口和局部变量窗口 &#xff08;2&#xff09;监视窗口 &#xff08;3&#xff09;调用堆栈 &#xff08;4&#xff09;查看汇编信息 &#xff08;5&#xff09;查看寄存…

Java零基础 - 算术运算符

哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c;我是喵手。 今天我要给大家分享一些自己日常学习到的一些知识点&#xff0c;并以文字的形式跟大家一起交流&#xff0c;互相学习&#xff0c;一个人虽可以走的更快&#xff0c;但一群人可以走的更远。 我是一名后…

Win11网络连接选项和蓝牙选项突然消失的解决办法

在设置或者开始栏里搜索“网络重置” 打开网络重置&#xff1a; 然后点击立即重置&#xff0c;之后按照系统提示操作即可

Mybatis总结--传参二

#叫做占位符 Mybatis是封装的JDBC 增强版 内部还是用的jdbc 每遇到一个#号 这里就会变为&#xff1f;占位符 一个#{}就是对应一个问号 一个占位符 用这个对象执行sql语句没有sql注入的风险 八、多个参数-使用Param 当 Dao 接口方法有多个参数&#xff0c;需要通过名称使…

猫头虎分享已解决Bug || 超时错误:TimeoutError: Request timed out after 30000ms.

博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题大全》 — 面试准备的宝典&#xff01;《IDEA开发秘籍》 — 提升你的IDEA技能&#xff01;《100天精通鸿蒙》 …

进程1——进程与线程——day09

今天&#xff0c;主要讲一下进程的一些基本概念和一些接口 首先是进程的基本概念&#xff1a; 1.进程: 程序&#xff1a;存放在外存中的一段数据组成的文件 进程&#xff1a;是一个程序动态执行的过程,包括进程的创建、进程的调度、进程的消亡 2.进程相关命令: 1.top 动态…

51单片机项目(34)——基于51单片机和esp8266的智能农业检测系统

1.设计要求 可以检测农业大棚中的温度、湿度、气压、光照等信息&#xff0c;可以检测土壤湿度&#xff0c;可以判断当前有没有下雨&#xff0c;能够将相关数据显示在OLED屏幕上。同时&#xff0c;使用esp8266wifi模块&#xff0c;将上述所有信息发送到手机APP显示。当温度、湿度…

SQL-Labs46关order by注入姿势

君衍. 四十六关 ORDER BY数字型注入1、源码分析2、rand()盲注3、if语句盲注4、时间盲注5、报错注入6、Limit注入7、盲注脚本 四十六关 ORDER BY数字型注入 请求方式注入类型拼接方式GET报错、布尔盲注、延时盲注ORDER BY $id 我们直接可以从界面中得知传参的参数为SORT&#x…

可变参数、Collections类

一、可变参数 定义&#xff1a;是一种特殊的形参&#xff0c;定义在方法、构造器的形参列表里 格式&#xff1a;数据类型...参数名称 特点&#xff1a;可以不传数据&#xff0c;也可以传一个或者多个数据给它&#xff0c;也可以传一个数组 好处&#xff1a;可以灵活接收数据…

刷题日记-Day2- Leedcode-977. 有序数组的平方,209. 长度最小的子数组,59. 螺旋矩阵 II-Python实现

刷题日记Day2 977 有序数组的平方209. 长度最小的子数组59. 螺旋矩阵 II 977 有序数组的平方 链接&#xff1a;https://leetcode.cn/problems/squares-of-a-sorted-array/description/ 给你一个按 非递减顺序 排序的整数数组 nums&#xff0c;返回 每个数字的平方 组成的新数组…

真Unity3D编辑器Editor二次开发

IMGUI Editor Label 改变颜色 分享一个很神奇的颜色 一开始这么写&#xff0c;以为不行的&#xff0c; private void OnGUI()(){GUILayout.Label("<colorred>name:</color>ffdasilufoi");//。。。。 } 结果这么写又好了&#xff0c; private GUIStyle m…

Vivado MIG ip核使用教程

Step 1 在ip catalog中搜索mig ip核并打开&#xff0c;检查硬件配置 Step 2 Step 3 选择对其他芯片类型的兼容性&#xff0c;若无此方面需求&#xff0c;可直接点击next Step 4 选择存储器类型 Step 5 配置DDR3芯片工作频率、用户时钟、mig ip核输入时钟、DDR3芯片类型…