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

在这里插入图片描述

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

  • 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://xiahunao.cn/news/2807139.html

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

相关文章

应用回归分析:贝叶斯回归

贝叶斯回归是一种统计方法&#xff0c;它利用贝叶斯定理来更新对回归参数的估计。这种方法不仅考虑了数据的不确定性&#xff0c;还考虑了模型参数的不确定性&#xff0c;为预测提供了一个更加全面的框架。在本文中&#xff0c;我们将深入探讨贝叶斯回归的基本概念、如何实现它…

项目分享,正在开发的一款多商户家政平台服务小程序【商户端】页面截图-技术栈为uniapp可生成H5/APP/小程序

页面展示 商户端首页 【工作台】 此部分的功能逻辑如下&#xff1a; 工作台&#xff1a;商户的操作中心 基本信息管理&#xff1a;商户端的工作台允许商户维护自己的基本信息&#xff0c;包括上传头像、设置店名、查看入驻时间以及选择店铺类型&#xff08;企业店铺或个人店…

http协议及httpd的使用

HTTP协议介绍 HTTP&#xff08;HyperText Transfer Protocol&#xff0c;超文本传输协议&#xff09;是一种用于分布式、协作式和超媒体信息系统的应用层协议。HTTP是万维网的数据通信的基础设计HTTP最初的目的是为了提供一种远距离共享知识的方式&#xff0c;借助多文档进行关…

单片机tsm32城市环境污染监测与实现

国内经济增速的持续保持不但加快了城市化建设的步伐&#xff0c;同时也使得更多的人口聚集到大城市中求发展&#xff0c;大量的人口对衣食住行等方面的需求使得这些城市环境的污染问题逐渐加剧。当前各级政府虽然对城市环境污染问题越来越重视&#xff0c;但是因缺乏监测手段而…

Qt 设置隐式加载dll路径

在c++中DLL的加载方式有两种,显式加载和隐式加载。 隐式加载 在程序从开始运行时,就会按照系统中一定的搜索路径,寻找动态库,找到就自动加载它,才能成功运行程序,这些步骤,是系统自动完成的。 显示加载 我们对动态库的调用,是在代码中直接使用LoadLibrary,或其他加载函…

用买糖果的方式来理解正交匹配追踪(OMP)算法

在信号处理领域&#xff0c;压缩感知&#xff08;Compressed Sensing&#xff09;是一种能够从远少于传统奈奎斯特采样定理所要求的样本数目中重建稀疏信号的技术。压缩感知的理论基础在于一个前提假设&#xff0c;即许多自然信号都含有稀疏的表示&#xff0c;换句话说&#xf…

神经网络系列---权重初始化方法

文章目录 权重初始化方法Xavier初始化&#xff08;Xavier initialization&#xff09;Kaiming初始化&#xff0c;也称为He初始化LeCun 初始化正态分布与均匀分布Orthogonal InitializationSparse Initializationn_in和n_out代码实现 权重初始化方法 Xavier初始化&#xff08;X…

Linux之ACL权限chmod命令

一. chmod命令 chmod命令来自英文词组change mode的缩写&#xff0c;其功能是改变文件或目录权限的命令。默认只有文件的所有者和管理员可以设置文件权限&#xff0c;普通用户只能管理自己文件的权限属性。 设置权限时可以使用数字法&#xff0c;亦可使用字母表达式&#xff0…

高考志愿选择辅助系统

高考志愿选择辅助系统 获取源码——》公主号&#xff1a;计算机专业毕设大全

STL - 并查集

1、并查集原理 在一些应用问题中&#xff0c;需要将n个不同的元素划分成一些不相交的集合&#xff1b;开始时&#xff0c;每个元素自成一个 单元素集合&#xff0c;然后按一定的规律将归于同一组元素的集合合并&#xff1b;在此过程中要反复用到查询某一 个元素归属于那个集合的…

java集合--List集合的基本用法

一、ArrayList集合 1.ArrayList集合的特点 2.ArrayList集合的一些方法 ①.add(Object element) 向列表的尾部添加指定的元素。 ②.size() 返回列表中的元素个数。 ③.get(int index) 返回列表中指定位置的元素&#xff0c;index从0开始。 public class Test {public static …

Beyond Compare4破解方法

方式一 第一种办法&#xff08;也是最有效的&#xff09; 删除C:\Users\用户名\AppData\Roaming\Scooter Software\Beyond Compare 4下的所有文件&#xff0c;重启Beyond Compare 4即可&#xff08;注意&#xff1a;用户名下的AppData文件夹有可能会被隐藏起来) 方式二 删…

光谱数据处理:1.特征波长优选的不同方法与Python实现

首先&#xff0c;我们要理解为什么要对“光谱数据进行特征波长优选”以及这是在干嘛&#xff0c;光谱数据可以想象成一长串的彩色条纹&#xff0c;每种颜色对应一个波长&#xff0c;就像彩虹一样。这些颜色的条纹代表了从某种物质&#xff08;比如植物、矿石或是食品&#xff0…

leet hot 100-1 两数之和

两数之和 原题链接思路代码 原题链接 leet hot 100-1 1. 两数之和 思路 可以把当前数字放到容器里面去 当我们遍历一个新的数字的时候 减一下与目标值的差 然后得到的结果在容器里面查看是否存在 时间复杂度O(n) 空间复杂度(n) 代码 class Solution { public:vector<…

!!!Python虚拟环境改名后的坑!!!!

搞了一晚上终于弄好这python虚拟环境的问题了&#xff01;真的是坑啊&#xff01; 本来用的纯python环境下的虚拟环境&#xff0c;一时心血来潮&#xff0c;把电脑重新装了一遍&#xff0c;虚拟环境的目录也改了一下&#xff0c;结果虚拟环境再vscode中是可以使用&#xff0c;…

Uniapp小程序开发-底部tabbar的开发思路

文章目录 前言一、uniapp 实现 tabbar二、图标使用网络图片后端返回tabbar信息uniapp方式中的setTabBarItem 总结 前言 记录uniapp 开发小程序的底部tabbar &#xff0c;这里讨论的不是自定义tabbar的情况。而是使用wx.setTabBarItem(Object object) 这个api的情况。关于custo…

通天星CMSV6 车载视频监控平台信息泄露漏洞

免责声明&#xff1a;文章来源互联网收集整理&#xff0c;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;所产生的一切不良后果与文章作者无关。该…

uni-app vue3 setup nvue中webview层级覆盖问题

核心就是这两行&#xff0c;&#x1f923;发现设置后不能点击了&#xff0c;这个玩意可能只能弹窗打开的时候动态的修改 position: static, zindex: 0onLoad(options > {loadWebview()})function loadWebview() {let pageInfo uni.getSystemInfoSync();width.value pageI…

抽象的java

Consider defining a bean of type org.springframework.mail.MailSender in your configuration. 报错原因&#xff1a; 第一个&#xff1a;未安装对应的依赖 第二个&#xff1a;对应配置问题 背景&#xff1a;用springboot-java完成邮箱发送 第一个问题解决方法&#xff1…

太阳能光伏电池模型参数辨识模型介绍

一、太阳能光伏电池模型参数辨识模型介绍 由于传统化石能源短缺问题日益严重&#xff0c;我国对新能源发展的重视提到了前所未有的高度。太阳能作为一种可再生能源&#xff0c;不会对环境造成污染&#xff0c;受到了越来越多的关注太阳能由于其储量丰富,无污染和无地域限制等优…