动态规划算法专题(一):斐波那契数列模型

news2024/9/30 10:04:37

目录

1、动态规划简介

2、算法实战应用【leetcode】

2.1 题一:第N个泰波那契数

2.1.1 算法原理

2.1.2 算法代码

2.1.3 空间优化原理——滚动数组

2.1.4 算法代码——空间优化版本

2.2 题二:三步问题

 2.2.1 算法原理

2.2.2 算法代码

2.3 题二:使用最小花费爬楼梯

2.3.1 算法原理【解法一】

2.3.2 算法代码【解法一】

2.3.3 算法原理【解法二】

​2.3.4 算法代码【解法二】

2.4 题三:解码方法

2.4.1 算法原理

2.4.2 算法代码

2.4.3 初始化技巧

2.4.4 算法代码——初始化技巧使用


1、动态规划简介

‌动态规划(Dynamic Programming, DP)是运筹学的一个分支,用于求解最优化问题。

在解决问题时,每次产生的子问题并不总是新问题,有些子问题被反复计算多次。动态规划算法(自底向上)正是利用了这种子问题的重叠性质,对每一个子问题只解一次,而后将其解保存在一个表格中,在以后尽可能多地利用这些子问题的解。

动态规划算法,一般包含以下几个主要步骤:

  1. 状态表示:dp表中的值所表示的含义。一般由 经验+题目要求 得出。对于一维的dp表,dp[i]通常有这两种表示形式(由经验得出):①:以i位置为结尾,......;②:以i位置为起点,......;而....就是需要结合题目来得出的。
  2. 状态转移方程:就是dp[i]是怎么来的。dp[i]通常由其相邻的其他状态经过公式计算得出,状态转移方程就是这个公式。
  3. 初始化:确保在填dp表的过程中不越界。
  4. 填表顺序为了在填写当前状态的时候,所需要的其他相邻状态已经计算得出了。
  5. 返回值:结合题目要求。

在此过程中,我认为确定状态表示是最重要的一点,而确定状态转移方程是最难的一点。 

2、算法实战应用【leetcode】

2.1 题一:第N个泰波那契数

. - 力扣(LeetCode)

2.1.1 算法原理

对于一维dp,状态表示通常由 经验+题目要求 得出,本题很明显,表示第i个泰波那契数的值;同时状态转移方程题目也是贴心的给出了。

  • 状态表示dp[i]:第i个泰波那契数的值
  • 状态转移方程:dp[i] = dp[i-1] + dp[i-2] + dp[i-3]
  • 初始化:dp[0] = 0; dp[1] = dp[2] = 1;
  • 填表顺序:从左往右
  • 返回值:dp[n]

2.1.2 算法代码

class Solution {
    public int tribonacci(int n) {
        if(n == 0) return 0;
        if(n == 1 || n == 2) return 1;
        int[] dp = new int[n + 1];
        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];
    }
}

2.1.3 空间优化原理——滚动数组

实际上,我们可以不用创建一个dp数组来存储数值,

通过“滚动数组”的方式,进行空间优化。

仅仅使用三个变量就可以解题,不必再开辟额外的空间。

2.1.4 算法代码——空间优化版本

class Solution {
    public int tribonacci(int n) {
        //动态规划 -> 空间优化
        if(n == 0) return 0;
        if(n == 1 || n == 2) return 1;
        int a = 0, b = 1, c = 1, d = 0;
        for(int i = 3; i <= n; i++) {
            d = a + b + c;
            //滚动操作
            a = b;
            b = c;
            c = d;
        }
        return d;
    }
}

2.2 题二:三步问题

. - 力扣(LeetCode)

 2.2.1 算法原理

  • 状态表示(经验+题目要求):dp[i]:到达第i个位置,一共有多少种方式
  • 状态转移方程:要到达i位置,可由移动到i-1/i-2/i-3位置的基础上,再移动三/二/一步就可到达,即:dp[i]=dp[i-1]+dp[i-2]+dp[i-3];
  • 初始化:dp[1]=1; dp[2]=2; dp[3]=4;
  • 填表顺序:从左往右

2.2.2 算法代码

class Solution {
    //dp[i]状态表示:到达i位置,一共有多少种方法
    public int waysToStep(int n) {
        //1e9 + 7:double类型
        int MOD = (int)1e9 + 7;
        if(n == 1 || n == 2) return n;
        int[] dp = new int[n + 1];
        //初始化
        dp[1] = 1; dp[2] = 2; dp[3] = 4;
        for(int i = 4; i < n + 1; i++) {
            //状态转移方程
            dp[i] = ((dp[i - 1] + dp[i - 2]) % MOD + dp[i - 3]) % MOD;
        }
        return dp[n];
    }
}

2.3 题二:使用最小花费爬楼梯

. - 力扣(LeetCode)

2.3.1 算法原理【解法一】

  • 状态表示dp[i]到达i位置,最小花费的金额(以i位置为结尾)
  • 状态转移方程:需要考虑,到达i-1位置和到达i-2位置最小花费的金额数,考虑完后,再进一步到达i位置。也就是说:dp[i]=min(dp[i-1]+cost[i-1],dp[i-2]+cost[i-2]);
  • 初始化:dp[0]=0; dp[1]=0;(防越界)
  • dp的建表顺序:从左至右
  • 返回值:dp[n]

2.3.2 算法代码【解法一】

class Solution {
    public int minCostClimbingStairs(int[] cost) {
        //解法一:
        int n = cost.length;
        //状态表示dp[i]:到达i位置时,最小花费
        int[] dp = new int[n + 1];
        //初始化
        dp[0] = dp[1] = 0;
        //填表顺序:从左往右
        for(int i = 2; i <= n; i++) {
            //状态转移方程
            dp[i] = Math.min(dp[i - 1] + cost[i - 1], dp[i - 2] +cost[i - 2]);
        }
        //返回值:dp[n]
        return dp[n];
    }
}

2.3.3 算法原理【解法二】

  • 状态表示dp[i]从i位置开始,到达楼顶,最小花费的金额(以i位置为起点)
  • 状态转移方程:需要考虑,i+1位置和i+2位置到达楼顶最小花费的金额数,再选择走一步还是走两步。也就是说:dp[i]=min(cost[i-1],dp[i-2])+cost[i];
  • 初始化:dp[i-1]=cost[i-1]; dp[i-2]=cost[i-2];(防越界)
  • dp的建表顺序:从右至左
  • 返回值:min(dp[0],dp[1])

​2.3.4 算法代码【解法二】

class Solution {
    public int minCostClimbingStairs(int[] cost) {
        int n = cost.length;
        //状态表示dp[i]:以i位置为起点,到达楼顶的最小花费
        int[] dp = new int[n];
        //初始化
        dp[n - 1] = cost[n - 1];
        dp[n - 2] = cost[n - 2];
        for(int i = n - 3; i >= 0; i--) {
            //填表顺序 -> 从右往左
            //状态转移方程 --> 自身花费+后面俩台阶的最小花费
            dp[i] = cost[i] +Math.min(dp[i + 1], dp[i + 2]);
        }
        //返回值,考虑 从第一个台阶出发 or 从第二个台阶出发
        return Math.min(dp[0], dp[1]);
    }
}

2.4 题三:解码方法

. - 力扣(LeetCode)

2.4.1 算法原理

  • 状态表示dp[i]:以i位置为结尾,解码方式的总数
  • 状态转移方程:s[i]单独解码(成功/失败)+s[i]/s[i-1]组合解码(成功/失败):dp[i]=dp[i-1]+dp[i-2];
  • 初始化:①:if(s[0]!='0') dp[0]=1; ②:dp[1] -->1. s[1]单独解码 & 2. s[1]与s[0]组合解码
  • 返回值:dp[n-1]

2.4.2 算法代码

class Solution {
    public int numDecodings(String s) {
        int n = s.length();
        char[] c = s.toCharArray();
        //状态表示dp[i]:以i位置为结尾,解码方式的总数
        int[] dp = new int[n];
        //初始化第一个位置
        if(c[0] != '0') dp[0]++;
        //当长度为1时
        if(n == 1) return dp[0];
        //初始化第二个位置
        if(c[0] != '0' && c[1] != '0') dp[1]++;//单独编码
        int t = ((c[0] - '0') * 10 + (c[1] - '0'));
        if(t >= 10 && t <= 26) dp[1]++;//组合编码
        for(int i = 2; i < n; i++) {
            //单独编码
            if(c[i] != '0') dp[i] += dp[i - 1];
            //组合编码
            int tt = ((c[i - 1] - '0') * 10 + (c[i] - '0'));
            if(tt >= 10 && tt <= 26) dp[i] += dp[i - 2];
        }
        return dp[n - 1];
    }
}

2.4.3 初始化技巧

在编写代码时,我们发现了一个问题, 我们在完成初始化工作时,非常的麻烦,初始化的代码占据了大量的篇幅,此时,我们可以使用初始化技巧 ---> dp数组在创建时多开辟一个节点。将初始化时的代码放进循环中(将要初始化的节点与普通节点一概而论),在填dp表的时候完成初始化。

不过,因为多开辟了一个位置,所以在循环填dp表时也需要注意以下问题:

  1. 与原数组的下标映射关系发生了改变
  2. 注意虚拟节点的值(多开辟的那一个节点),虚拟节点的值,要保证后面计算得到的结果是正确的。

在本题中,虚拟节点dp[0]的值应该为1;

原因:

当s[1]可与s[0]组合解码时,会这样计算:dp[2]+=d[2-2](s中的i为1,映射到新dp中i为2)
因为能够组合解码,所以虚拟节点dp[0]的值应为1

2.4.4 算法代码——初始化技巧使用

class Solution {
    public int numDecodings(String s) {
        //初始化技巧 --> 优化
        int n = s.length();
        char[] c = s.toCharArray();
        //状态表示dp[i + 1]:以i位置为结尾,解码方式的总数
        int[] dp = new int[n + 1];
        //处理虚拟节点里的值
        dp[0] = 1;
        //初始化第一个位置
        if(c[0] != '0') dp[1]++;
        for(int i = 2; i < n + 1; i++) {
            //单独编码
            if(c[i - 1] != '0') dp[i] += dp[i - 1];
            //组合编码
            int t = ((c[i - 1 - 1] - '0') * 10 + (c[i - 1] - '0'));
            if(t >= 10 && t <= 26) dp[i] += dp[i - 2];
        }
        return dp[n];
    }
}

END

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

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

相关文章

数据集-目标检测系列-鼠检测数据集 mouse >> DataBall

数据集-目标检测系列-鼠检测数据集 mouse >> DataBall 数据集-目标检测系列-老鼠检测数据集 mouse 数据量&#xff1a;6k 想要进一步了解&#xff0c;请联系。 DataBall 助力快速掌握数据集的信息和使用方式&#xff0c;会员享有 百种数据集&#xff0c;持续增加中。…

高效互动随心畅谈语聊大厅语音聊天系统小程序源码

高效互动随心畅谈 —— 语聊大厅语音聊天系统全解析 &#x1f389; 初识语聊大厅&#xff1a;语音社交的新大陆 在这个快节奏的数字时代&#xff0c;文字聊天似乎已经无法满足我们对即时互动和真实情感交流的需求。而“高效互动随心畅谈语聊大厅语音聊天系统”的出现&#xff…

TiDB 在线打标签实现副本调度应用实践

作者&#xff1a; 数据源的TiDB学习之路 原文来源&#xff1a; https://tidb.net/blog/4e14596a 案例背景 某原有系统为虚拟机环境部署&#xff0c;整体性能不满足预期。为提升集群整体性能&#xff0c;计划分阶段采购物理机&#xff0c;并以扩缩容的方式逐渐把物理机添加到…

如何从 Windows 11/10/8.1/8/7 中恢复已删除的视频

不小心删除了视频或格式化了 SD 卡/硬盘&#xff1f;没有备份已删除的视频&#xff1f;不要担心&#xff0c;我们有一个解决方案 可以恢复 Windows 11、10 中已删除的视频并处理这种可怕的情况。 但是&#xff0c;在详细介绍如何恢复已删除的视频和视频恢复应用程序之前&#…

【AI论文精读1】针对知识密集型NLP任务的检索增强生成(RAG原始论文)

目录 一、简介一句话简介作者、引用数、时间论文地址开源代码地址 二、摘要三、引言四、整体架构&#xff08;用一个例子来阐明&#xff09;场景例子&#xff1a;核心点&#xff1a; 五、方法 &#xff08;架构各部分详解&#xff09;5.1 模型1. RAG-Sequence Model2. RAG-Toke…

docker安装Portainer CE

docker安装Portainer CE 教程 1、简介 Portainer 是一款开源的容器管理工具&#xff0c;旨在帮助用户更轻松地管理 Docker 环境。无论您是 Docker 新手还是经验丰富的开发人员&#xff0c;Portainer 都提供了直观的用户界面&#xff0c;使您能够方便地创建、部署和监控容器。…

抖去推数字人---技术本地服务器技术开发步骤

AI数字人制作流程大致可以分为以下几个关键步骤&#xff1a; 虚拟形象设计&#xff1a;设计虚拟人物的外观和特征&#xff0c;可以手工完成&#xff0c;也可以利用计算机图形学和机器学习算法自动生成。 驱动/动作捕捉&#xff1a;使用动作捕捉技术记录真实演员的动作&#xf…

Rust Web自动化Demo

1.新建项目 使用RustCover新建项目&#xff0c;目录如下&#xff1a; Cargo.toml文件 [package] name "Demo" version "0.1.0" edition "2021"[dependencies] tokio { version "1", features ["full"] } thirtyfour…

【Android 源码分析】Activity短暂的一生 -- 目录篇 (持续更新)

1. 前言 忽然有一天&#xff0c;我想要做一件事&#xff1a;去代码中去验证那些曾经被“灌输”的理论。                                                                                  …

光电开关应用设计讨论

一. 检测原理 光电开关由一个发光管和一个光敏二极管或三极管组成。其原理很简单&#xff0c;发光二极管发出的光传递到光敏三极管&#xff0c;再转换成电信号。如图1所示&#xff0c;LED为发射端&#xff0c;正向电流IF越大&#xff0c;发射光的强度就越大&#xff1b;PT为接…

【小程序】微信小程序课程 -4 项目实战

目录 1、 效果图 2、创建项目 2.1 创建小程序端 2.1.1 先创建纯净项目 2.1.2 删除components 2.1.4 删除app.json红色部分 2.1.5 删除index.json红色部分 2.1.6 删除index.wxss全部内容 2.1.7 删除index.wxml全部内容 2.1.8 app.json创建4个页面 2.1.9 app.json添加…

【拥抱AIGC】通义灵码网络代理配置

在公司网络环境下&#xff0c;无法访问公共网络时&#xff0c;可在插件端配置网络代理后使用通义灵码。 配置网络代理 公司网络通常使用 HTTP 代理服务器在网络流量发送到目标位置之前进行拦截&#xff0c;以便检测可疑流量或者限制进入公司内网络的内容。如果你使用的公司网…

【Python报错已解决】 NameError: name ‘scio‘ is not defined

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 专栏介绍 在软件开发和日常使用中&#xff0c;BUG是不可避免的。本专栏致力于为广大开发者和技术爱好者提供一个关于BUG解决的经…

Star 3w+,向更安全、更泛化、更云原生的 Nacos3.0 演进

作者&#xff1a;席翁 Nacos 社区刚刚迎来了 Star 突破 30000 的里程碑&#xff0c;从此迈上了一个新的阶段。感谢大家的一路支持、信任和帮助&#xff01; Nacos /nɑ:kəʊs/是 Dynamic Naming and Configuration Service 的首字母简称&#xff0c;定位于一个更易于构建云原…

c++就业磁盘链式b树与b+树

linux上/a.out启动一个进程 最上面是内核 在heap里 一个节点只想左子树 另一个节点指向右子树 找到根节点 对比找 磁盘寻址过程 对比次数多了之后 找下一个节点次数变多 磁盘面-道-区 一个节点4k 不断在磁盘上寻址 开叉 中间存储数据 b树 所有的叶子节点在同一层 b树 所有节点…

Llama 3.1 技术研究报告-7

7.6 图像识别结果 我们评估了 Llama 3 在⼀系列任务上的图像理解能⼒&#xff0c;这些任务涵盖了⾃然图像理解、⽂本理解、图表理解和多模态推理&#xff1a; MMMU (Yue 等⼈&#xff0c;2024a) 是⼀个具有挑战性的数据集&#xff0c;⽤于多模态推理&#xff0c;模型需要理解…

tesseract:一个.Net版本的开源OCR项目

推荐一个.Net版本的开源OCR项目&#xff0c;方便我们在项目中集成OCR功能。 01 项目简介 tesseract是针对Tesseract-OCR&#xff08;C&#xff09;引擎封装的.NET版本&#xff0c;支持超过100种语言的文本识别&#xff0c;使得.NET开发者能够轻松地利用Tesseract的强大功能&a…

LLM基础概念-训练集

数据集 训练集(Training Set) 用来训练模型的数据。就像给学生提供教材一样&#xff0c;训练集帮助模型学习如何从输入数据预测出正确的结果。 验证集(Validation Set) 这个数据集用来检查模型在训练过程中的表现。它帮助我们调整模型训练参数的设置&#xff0c;以确保模型不…

《基于 Spring Boot 的健身房管理系统功能介绍》

一、系统概述 本健身房管理系统基于 Spring Boot 框架开发&#xff0c;旨在为健身房提供一套高效、便捷的管理解决方案。系统涵盖了会员卡查询、会员管理、员工管理、器材管理和课程管理等核心功能&#xff0c;帮助健身房实现全面的数字化管理。 二、会员卡查询 用户可以通过输…

影响 Linux、Unix 系统的 CUPS 漏洞可导致 RCE

在经过大量炒作和第三方过早泄露信息之后&#xff0c;安全研究员 Simone Margaritelli 公布了有关通用 UNIX 打印系统 (CUPS) 中的四个零日漏洞的详细信息。 这些漏洞可被远程、未经身份验证的攻击者滥用&#xff0c;在易受攻击的 Linux 和类 Unix 系统上实现代码执行。 CUPS…