代码随想录刷题|动态规划理论基础 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];
    }
}


相关文章
|
1月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
73 2
|
1月前
|
算法 Python
【Leetcode刷题Python】 LeetCode 2038. 如果相邻两个颜色均相同则删除当前颜色
本文介绍了LeetCode 2038题的解法,题目要求在一个由'A'和'B'组成的字符串中,按照特定规则轮流删除颜色片段,判断Alice是否能够获胜,并提供了Python的实现代码。
36 3
|
1月前
|
Python
【Leetcode刷题Python】50. Pow(x, n)
本文介绍了LeetCode第50题"Pow(x, n)"的解法,题目要求实现计算x的n次幂的函数,文章提供了递归分治法的详细解析和Python实现代码。
15 1
|
1月前
|
Python
【Leetcode刷题Python】LeetCode 478. 在圆内随机生成点
本文介绍了LeetCode 478题的解法,题目要求在给定圆的半径和圆心位置的情况下实现在圆内均匀随机生成点的功能,并提供了Python的实现代码。
16 1
|
1月前
|
算法 Python
【Leetcode刷题Python】295. 数据流的中位数
本文介绍了一种使用Python实现的数据结构,用以支持数据流中添加整数并返回当前所有元素的中位数,通过排序列表来计算中位数。
14 1
|
1月前
|
算法 Python
【Leetcode刷题Python】73. 矩阵置零
本文介绍了LeetCode第73题的解法,题目要求在给定矩阵中将所有值为0的元素所在的行和列全部置为0,并提供了一种原地算法的Python实现。
17 0
【Leetcode刷题Python】73. 矩阵置零
|
1月前
|
Python
【Leetcode刷题Python】1467. 两个盒子中球的颜色数相同的概率
本文介绍了LeetCode第50题"Pow(x, n)"的解法,题目要求实现计算x的n次幂的函数,文章提供了递归分治法的详细解析和Python实现代码。
18 0
|
1月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
41 6
|
1月前
|
Python
【Leetcode刷题Python】剑指 Offer 26. 树的子结构
这篇文章提供了解决LeetCode上"剑指Offer 26. 树的子结构"问题的Python代码实现和解析,判断一棵树B是否是另一棵树A的子结构。
35 4
|
1月前
|
索引 Python
【Leetcode刷题Python】从列表list中创建一颗二叉树
本文介绍了如何使用Python递归函数从列表中创建二叉树,其中每个节点的左右子节点索引分别是当前节点索引的2倍加1和2倍加2。
36 7