代码随想录刷题|动态规划理论基础 LeetCode 509. 斐波那契数 70. 爬楼梯 746. 使用最小花费爬楼梯(上)

简介: 代码随想录刷题|动态规划理论基础 LeetCode 509. 斐波那契数 70. 爬楼梯 746. 使用最小花费爬楼梯

动态规划理论基础


 递归算法是一种穷举的展示,任何数学递推公式都可以直接转换成递归算法,递归算法是当前层需要上一层的准备。递归算法冗余计算的增长是爆炸性的。如果编译器的递归模拟算法要是能够保留一个预先算出的值的表,而对已经解过的子问题不用再进行递归调用,那么这种指数式的爆炸性的增长就可以避免


       动态规划就是这样的算法


       如果某一个问题可以解决很多重叠问题,那么使用动态规划是最有效的解决办法


       动态规划中的每一个状态是由上一个状态推导出来的,这和贪心算法是不同的,贪心算法是没有状态推导的过程,贪心算法都是在当前局部选取最优解。所以贪心算法解决不了动态规划的问题


       动态规划五部曲:

               1、确定dp数组(dp table)及下标的含义

               2、确定递推公式

               3、初始化dp数组

               4、确定遍历顺序

               5、举例推导dp数组


509. 斐波那契数

题目链接:力扣

思路

下面是对递归的优化:

用表优化递归


       从这道题目可以看出,递归算法是需要不断向下进行计算,进行压栈,比如要算F(5),那就要先算出F(4)和F(3),就要先算出F(3)、F(2)、F(2)、F(1)……,这里可以看到是有重复的运算的


       那我们可以先创建一个数组(表),能够保留一个预先算出的值,这样就可以不用,这样就避免了重复的运算。已经算过的F(n)就不用再进行重复的计算


       根据动态规划五部曲:

               1、确定dp数组(dp table)及下标的含义

                       第i个下标存储的是F(i)斐波那契的值

               2、确定递推公式

                       递推公式是:dp[i] = dp[i - 1]+dp[i - 2]

               3、初始化dp数组

                       dp[0] = 0;

                       dp[1] = 1;

               4、确定遍历顺序

                       从递推公式可以得出,后面的数字依赖前面的数字,所以是从前向后遍历

               5、举例推导dp数组

                       0 1 1 2 3 5 8 13 21 34 55


下面是对表的优化:


       得出动态对话的代码之后会发现,代码的内存消耗比较大,我们发现处理,中间过程中的数字不都用,但是还是占用了内存,这样的内存占用时浪费的。比如F(10)。dp数组为:0 1 1 2 3 5 8 13 21 34 55。但是只会返回55,前面的都不用


       其实只需要维护两个数值就可以了,不需要记录整个数列。在整个遍历的过程中

       dp[0] = dp[i - 1]

       dp[1] = dp[i - 2]

       最后循环结束的时候dp[1]就是我们要的值


斐波那契数

递归思路


b7b35f0672284e7695c2faf302437bd4.png

class Solution {
    public int fib(int n) {
        // 终止条件
        if (n <= 1) {
            return n;
        }
        return fib(n - 1) + fib(n - 2);
    }
}

动态规划

cf650c3467c74b769e18d9f93669c07d.png

class Solution {
    public int fib(int n) {
        if (n == 0) {
            return 0;
        }
        if (n == 1) {
            return 1;
        }
        // 创建数组
        int[] dp = new int[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];
    }
}

动态规划(优化数组)

72e20b3d7fba4b5da28b8ba765449437.png

class Solution {
    public int fib(int n) {
        if (n <= 1) {
            return n;
        }
        // 创建数组
        int[] dp = new int[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];
    }
}


相关文章
|
2月前
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
3月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
113 2
|
19天前
|
机器学习/深度学习 人工智能 自然语言处理
280页PDF,全方位评估OpenAI o1,Leetcode刷题准确率竟这么高
【10月更文挑战第24天】近年来,OpenAI的o1模型在大型语言模型(LLMs)中脱颖而出,展现出卓越的推理能力和知识整合能力。基于Transformer架构,o1模型采用了链式思维和强化学习等先进技术,显著提升了其在编程竞赛、医学影像报告生成、数学问题解决、自然语言推理和芯片设计等领域的表现。本文将全面评估o1模型的性能及其对AI研究和应用的潜在影响。
16 1
|
2月前
|
数据采集 负载均衡 安全
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
本文提供了多个多线程编程问题的解决方案,包括设计有限阻塞队列、多线程网页爬虫、红绿灯路口等,每个问题都给出了至少一种实现方法,涵盖了互斥锁、条件变量、信号量等线程同步机制的使用。
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
|
3月前
|
Python
【Leetcode刷题Python】50. Pow(x, n)
本文介绍了LeetCode第50题"Pow(x, n)"的解法,题目要求实现计算x的n次幂的函数,文章提供了递归分治法的详细解析和Python实现代码。
26 1
|
3月前
|
算法 Python
【Leetcode刷题Python】73. 矩阵置零
本文介绍了LeetCode第73题的解法,题目要求在给定矩阵中将所有值为0的元素所在的行和列全部置为0,并提供了一种原地算法的Python实现。
32 0
【Leetcode刷题Python】73. 矩阵置零
|
3月前
|
Python
【Leetcode刷题Python】1467. 两个盒子中球的颜色数相同的概率
本文介绍了LeetCode第50题"Pow(x, n)"的解法,题目要求实现计算x的n次幂的函数,文章提供了递归分治法的详细解析和Python实现代码。
40 0
|
3月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
56 6
|
3月前
|
Python
【Leetcode刷题Python】剑指 Offer 26. 树的子结构
这篇文章提供了解决LeetCode上"剑指Offer 26. 树的子结构"问题的Python代码实现和解析,判断一棵树B是否是另一棵树A的子结构。
50 4
|
3月前
|
索引 Python
【Leetcode刷题Python】从列表list中创建一颗二叉树
本文介绍了如何使用Python递归函数从列表中创建二叉树,其中每个节点的左右子节点索引分别是当前节点索引的2倍加1和2倍加2。
56 7