【动态规划】是泰波那契数,不是斐波那契数

2023-10-11 12:44

本文主要是介绍【动态规划】是泰波那契数,不是斐波那契数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

在这里插入图片描述

Problem: 1137. 第 N 个泰波那契数

文章目录

  • 题目解读
  • 解题方法
    • dp动态规划
    • 迭代优化✔
  • 复杂度
  • Code

题目解读

首先我们来解读一下本题的意思🔍

1.jpg

  • 相信读者在看到【泰波那契数】的时候,不禁会联想到【斐波那契数】,它们呢是一对孪生兄弟,这个 泰波那契数 相当于是 斐波那契数 的加强版
  • 我们首先可以来看到这个递推公式Tn+3 = Tn + Tn+1 + Tn+2,读者可能看不太懂,我们将其做一个转换为Tn = Tn-1 + Tn-2 + Tn-3,即把所有n都统一-3那么第N个泰波那契数就等于前面3个泰波那契数的和

2.jpg

  • 看到上面的T3,就是前3个数的和等于2,以此类推T4就是T1 + T2 + T3 = 4

解题方法

看完了上面对于题目的分析之后,下面我将介绍两种解法

dp动态规划

首先的话就是本题需要掌握的重点即【动态规划】的解法,我们要分成五步去求解

  1. 状态表示
  • 首先读者要清楚的是在求解动态规划的题目时,都是需要一个dp数组的,那么对于【状态表示】的含义就是dp表里的值所表示的含义

3.jpg

那这个状态表示是怎么来的呢?

① 第一个呢就是按照题目要求来,dp[i]表示的就是第i个泰波那契数列的值

② 第二个呢则是需要读者有丰富的刷题经验,可以读完题目之后就得出对应的结果

③ 第三个呢则是在分析问题的过程中,发现重复的子问题

如果读者之前有接触过类似的动态规划问题的话,就会看到一些题解里讲:这道题的 状态表示 是怎样的,然后就直接讲本题的 状态表示方程,根本没有说这道题的状态表示是怎么来的。这个得来的过程我会在其他动态规划的题目中进行讲解

👉 所以读者在解类似的问题时一定要知道下面的这个【状态表示方程】是怎么来的,做到 “ 知其然,知其所以然 ”


  1. 状态表示方程
  • 那么本题的状态表示方程为dp[i] = dp[i - 1] + dp[i - 2] + dp[i - 3]
  1. 初始化
  • 在清楚【状态表示方程】该如何写之后,我们要去做的就是对这个dp数组做一个初始化的工作。看到下面的这个dp数组,如果在一开始我们的下标取值就到0的话,那么i - 1i - 2i - 3这些就会造成 越界

4.jpg

  • 因此我们要给这个dp数组去做一个初始化,具体的就是对前三个数据即dp[0]dp[1]dp[2]分别初始化为【0】【1】【1】,那我们在后面遍历计算的时候就只需要从下标为3的位置开始即可
 dp[0] = 0, dp[1] = dp[2] = 1;
  1. 填表顺序
  • 接下去的话就是把dp数组按照 状态表示方程 给填充好即可
for(int i = 3; i <= n; ++i)
{// 状态转移方程dp[i] = dp[i - 1] + dp[i - 2] + dp[i - 3];
}
  1. 返回值
  • 最后一块我们处理返回值,根据题目要求我们是要返回【第 n 个泰波那契数 Tn 的值】,所以直接 return dp[n] 即可

但是呢,若只考虑上面的这一些,在提交的时候是会出现越界的情况,因为在题目中给出的n的范围为[0, 37],因此对于dp[0]还好说,但对于后面的数据就会出现越界的情况

5.jpg

因此我们还需要去考虑一些边界的问题

// 边界问题处理
if(n == 0)  return 0;
if(n == 1 || n == 2)    return 1;

👉 整体代码会在最后给出

迭代优化✔

看完上面这一种,我们再来看看其是否可以去做一个优化

  • 如果读者有接触过迭代算法的话,应该很快能想到本题的思路,因为是三个三个去做的累加,所以我们在这里可以定义三个变量abc,它们累加后的值可以放到变量d

6.jpg

  • 因此在累加完一轮之后,我们就需要去做一个迭代的操作
a = b; b = c; c = d;

7.jpg

  • 那么在最后我们所需要返回的值就是这个d
return d;

复杂度

  • 时间复杂度:

对于第一种dp的解法,其时间复杂度为 O ( n ) O(n) O(n),而对于第二种迭代的解法时间复杂度为 O ( 1 ) O(1) O(1)

  • 空间复杂度:

对于第一种dp的解法,其空间复杂度为 O ( n ) O(n) O(n),而对于第二种迭代的解法时间复杂度为 O ( 1 ) O(1) O(1)

👉 所以就这么对比下来迭代优化的方法还是值得大家去掌握的

Code

首先是第一种dp动态规划的解法

class Solution {
public:int tribonacci(int n) {// 边界问题处理if(n == 0)  return 0;if(n == 1 || n == 2)    return 1;// 1.创建dp表vector<int> dp(n + 1);// 2.初始化dp[0] = 0, dp[1] = 1, dp[2] = 1;// 3.填表for(int i = 3; i <= n; ++i){// 状态转移方程dp[i] = dp[i - 1] + dp[i - 2] + dp[i - 3];}// 4.返回值return dp[n];}
};

然后的话是第二种利用迭代优化的方法

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;}
};

在这里插入图片描述

这篇关于【动态规划】是泰波那契数,不是斐波那契数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/188125

相关文章

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可

Mybatis嵌套子查询动态SQL编写实践

《Mybatis嵌套子查询动态SQL编写实践》:本文主要介绍Mybatis嵌套子查询动态SQL编写方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、实体类1、主类2、子类二、Mapper三、XML四、详解总结前言MyBATis的xml文件编写动态SQL

SpringBoot实现Kafka动态反序列化的完整代码

《SpringBoot实现Kafka动态反序列化的完整代码》在分布式系统中,Kafka作为高吞吐量的消息队列,常常需要处理来自不同主题(Topic)的异构数据,不同的业务场景可能要求对同一消费者组内的... 目录引言一、问题背景1.1 动态反序列化的需求1.2 常见问题二、动态反序列化的核心方案2.1 ht

golang实现动态路由的项目实践

《golang实现动态路由的项目实践》本文主要介绍了golang实现动态路由项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习... 目录一、动态路由1.结构体(数据库的定义)2.预加载preload3.添加关联的方法一、动态路由1

Python Selenium动态渲染页面和抓取的使用指南

《PythonSelenium动态渲染页面和抓取的使用指南》在Web数据采集领域,动态渲染页面已成为现代网站的主流形式,本文将从技术原理,环境配置,核心功能系统讲解Selenium在Python动态... 目录一、Selenium技术架构解析二、环境搭建与基础配置1. 组件安装2. 驱动配置3. 基础操作模

慢sql提前分析预警和动态sql替换-Mybatis-SQL

《慢sql提前分析预警和动态sql替换-Mybatis-SQL》为防止慢SQL问题而开发的MyBatis组件,该组件能够在开发、测试阶段自动分析SQL语句,并在出现慢SQL问题时通过Ducc配置实现动... 目录背景解决思路开源方案调研设计方案详细设计使用方法1、引入依赖jar包2、配置组件XML3、核心配

springboot使用Scheduling实现动态增删启停定时任务教程

《springboot使用Scheduling实现动态增删启停定时任务教程》:本文主要介绍springboot使用Scheduling实现动态增删启停定时任务教程,具有很好的参考价值,希望对大家有... 目录1、配置定时任务需要的线程池2、创建ScheduledFuture的包装类3、注册定时任务,增加、删

SpringBoot基于配置实现短信服务策略的动态切换

《SpringBoot基于配置实现短信服务策略的动态切换》这篇文章主要为大家详细介绍了SpringBoot在接入多个短信服务商(如阿里云、腾讯云、华为云)后,如何根据配置或环境切换使用不同的服务商,需... 目录目标功能示例配置(application.yml)配置类绑定短信发送策略接口示例:阿里云 & 腾

MySQL中动态生成SQL语句去掉所有字段的空格的操作方法

《MySQL中动态生成SQL语句去掉所有字段的空格的操作方法》在数据库管理过程中,我们常常会遇到需要对表中字段进行清洗和整理的情况,本文将详细介绍如何在MySQL中动态生成SQL语句来去掉所有字段的空... 目录在mysql中动态生成SQL语句去掉所有字段的空格准备工作原理分析动态生成SQL语句在MySQL