算法通关村第十九关——动态规划高频问题(白银)

前言

摘自:代码随想录

动态规划五部曲:

  1. 确定dp数组(dp table)及其下标的含义
  2. 确定递推公式
  3. 初始化dp数组
  4. 确定遍历顺序
  5. 举例推导dp数组

1 最少硬币数

leetcode 322. 零钱兑换

动规五部曲分析如下:

  1. 确定dp数组以及下标的含义

dp[j]:凑足总额为 j 所需钱币的最少个数为dp[j]

  1. 确定递推公式

凑足总额为j - coins[i]的最少个数为dp[j - coins[i]],

那么只需要加上一个钱币coins[i]即dp[j - coins[i]] + 1就是dp[j](考虑coins[i])

所以dp[j] 要取所有 dp[j - coins[i]] + 1 中最小的。

递推公式:dp[j] = min(dp[j - coins[i]] + 1, dp[j]);

  1. dp数组如何初始化

首先凑足总金额为0所需钱币的个数一定是0,那么dp[0] = 0;

其他下标对应的数值呢?

考虑到递推公式的特性,dp[j]必须初始化为一个最大的数,否则就会在min(dp[j - coins[i]] + 1, dp[j])比较的过程中被初始值覆盖。

所以下标非0的元素都是应该是最大值。

int[] dp = new int[amount + 1];
// 往数组dp里面填充某个数,这里选择amount+1,就是最大的值
Arrays.fill(dp, amount+1);
dp[0] = 0;
  1. 确定遍历顺序

有两种方式:

第一种:外循环遍历金额,内循环遍历硬币面额。

第二种:外循环遍历硬币面面额,内循环遍历金额。

这两种遍历顺序对应的意义如下:

  1. 外循环遍历金额,内循环遍历硬币面额:

    这种遍历顺序的意义是在计算找零过程中,我们首先考虑金额的变化,然后再考虑不同的硬币面额。

    也就是说,我们固定一个金额,尝试使用不同的硬币面额来找零。这样做的好处是可以利用之前已经计算出来的金额的最少硬币数,快速得到当前金额的最优解。由于金额是从小到大递增的,所以我们在计算每个金额的最优解时,可以利用前面较小金额的最优解已经被计算出来的特点。

// 遍历金额
for (int i = 1; i <= amount; i++) {
    // 遍历硬币面额
    for (int j = 0; j < coins.length; j++) {
        if (coins[j] <= i) {
            dp[i] = Math.min(dp[i], dp[i - coins[j]] + 1);
        }
    }
}
  1. 外循环遍历硬币面额,内循环遍历金额:

    这种遍历顺序的意义是在计算找零过程中,我们首先考虑不同的硬币面额,然后再考虑不同的金额。

    也就是说,我们固定一个硬币面额,尝试在不同的金额下进行找零。这样做的好处是可以保证我们将所有可能的硬币面额都考虑到,并且在计算每个金额的最优解时,可以利用之前已经计算出来的较小金额的最优解。由于硬币面额是从小到大递增的,所以我们在计算每个金额的最优解时,可以利用之前较小硬币面额的最优解已经被计算出来的特点。

// 遍历硬币面额
for (int coin : coins){
    // 遍历金额
    for (int i = 1; i <= amount; i++) {
        if(coin <= i){
            dp[i] = Math.min(dp[i], dp[i - coin] + 1);
        }
    }
}

全部代码如下:

第一种:

class Solution {
    public int coinChange(int[] coins, int amount) {
        // 初始化dp数组
        int[] dp = new int[amount + 1];
        Arrays.fill(dp, amount + 1);
        dp[0] = 0;

        // 遍历金额
        for (int i=1; i <= amount; i++) {
            // 遍历硬币面额
            for (int j=0; j < coins.length; j++){
                if(coins[j] <= i){
                    dp[i] = Math.min(dp[i], dp[i - coins[j]] + 1);
                }
            }
        }
            
            
        return dp[amount] > amount ? -1 : dp[amount];
    }
}

第二种:

class Solution {
    public int coinChange(int[] coins, int amount) {
        // 初始化dp数组
        int[] dp = new int[amount + 1];
        Arrays.fill(dp, amount + 1);
        dp[0] = 0;

        // 遍历硬币面额
        for (int coin : coins){
            // 遍历金额
            for (int i = 1; i <= amount; i++) {
                if(coin <= i){
                    dp[i] = Math.min(dp[i], dp[i - coin] + 1);
                }
            }
        }
            
        return dp[amount] > amount ? -1 : dp[amount];
    }
}

2 最长连续递增子序列

leetcode 674. 最长连续递增序列

动规五部曲分析如下:

  1. 确定dp数组以及下标的含义

dp数组:表示以当前元素为结尾的最长连续递增序列的长度。

dp[i]表示以nums[i]为结尾的最长连续递增序列的长度。

  1. 确定递推公式

如果nums[i] > nums[i-1],则dp[i] = dp[i-1] + 1;否则dp[i] = 1。

  1. dp数组如何初始化

我们将dp数组的所有元素初始化为1,因为每个元素都可以作为一个单独的递增序列。

  1. 确定遍历顺序

从第二个元素开始遍历:

for(int i=0; i < nums.length; i++){
    if(i > 0 && nums[i] > nums[i-1]){
        dp[i] = dp[i-1] + 1;
    }else{
        dp[i] = 1;
    }
}
  1. 举例说明

举例说明:给定数组nums = [1, 3, 5, 4, 7]。

遍历过程如下:

  • 对于nums[1] = 3,nums[0] = 1 < nums[1],所以dp[1] = dp[0] + 1 = 2。
  • 对于nums[2] = 5,nums[1] = 3 < nums[2],所以dp[2] = dp[1] + 1 = 3。
  • 对于nums[3] = 4,nums[2] = 5 > nums[3],所以dp[3] = 1。
  • 对于nums[4] = 7,nums[3] = 4 < nums[4],所以dp[4] = dp[3] + 1 = 2。

最终的最长连续递增序列的长度为dp数组的最大值,即为3。

最后代码如下:

class Solution {
    public int findLengthOfLCIS(int[] nums) {
        if (nums == null || nums.length == 0) {
            return 0;
        }

        int[] dp = new int[nums.length];
        dp[0] = 1;

        for(int i=1; i < nums.length; i++){
            if(nums[i] > nums[i-1]){
                dp[i] = dp[i-1] + 1;
            }else{
                dp[i] = 1;
            }
        }

        return Arrays.stream(dp).max().getAsInt();
    }
}

不是使用stream的方式:

class Solution {
    public int findLengthOfLCIS(int[] nums) {
        if (nums == null || nums.length == 0) {
            return 0;
        }

        int[] dp = new int[nums.length];
        dp[0] = 1;
        int maxLength = 1;

        for(int i=1; i < nums.length; i++){
            if(nums[i] > nums[i-1]){
                dp[i] = dp[i-1] + 1;
            }else{
                dp[i] = 1;
            }
            maxLength = Math.max(maxLength, dp[i]);
        }

        return maxLength;
    }
}

还可以得到dp[i],再遍历一遍得到最大值,这就不写了

3 最长递增子序列

leetcode 300. 最长递增子序列

  1. 确定dp数组(dp table)及其下标的含义:

    • dp数组:dp[i] 表示以第i个数字结尾的最长递增子序列的长度。
    • 下标的含义:dp[i] 表示以第i个数字结尾的最长递增子序列的长度。
  2. 确定递推公式:

    • 如果nums[i] > nums[j],则:dp[i] = max(dp[i], dp[j] + 1)。

    为啥呢??

    这里的i和j表示数组nums的索引。具体来说,i表示当前遍历到的元素的索引,而j表示在i之前的元素的索引。

    当我们遍历到第i个元素时,我们需要寻找在i之前的元素中比nums[i]小的元素。这样,我们就可以利用这个小于nums[i]的元素来构成一个更长的递增子序列。

    所以,当nums[i] > nums[j]时,表示nums[i]比nums[j]大,我们可以将以j结尾的最长递增子序列的长度加1,然后与以i结尾的最长递增子序列的长度进行比较,取较大的值作为以i结尾的最长递增子序列的长度。也就是递推公式中的 dp[i] = max(dp[i], dp[j] + 1)

  3. 初始化dp数组:

    • 初始时,dp数组中的每个元素都设为1,因为最短的递增子序列长度为1。
  4. 确定遍历顺序:

    • 外层循环遍历数组nums,从左到右依次计算dp[i]的值。
    • 内层循环遍历数组nums,从数组开始到i的位置,寻找前面的数字nums[j]是否小于nums[i],如果是,则根据递推公式更新dp[i]的值。
  5. 举例推导dp数组:

    如果nums[i] > nums[j],则dp[i] = max(dp[i], dp[j] + 1)。

    逐个元素计算dp[i]的值:

    • 当i = 1时,nums[i] = 9,此时没有比9小的元素,所以以9结尾的最长递增子序列长度仍为1。

    nums: 10 9 2 5 3 7 101 18
    dp: 1 1 1 1 1 1 1 1

    • 当i = 2时,nums[i] = 2,此时在2之前有9和10两个元素,都比2大,所以以2结尾的最长递增子序列长度仍为1。

    nums: 10 9 2 5 3 7 101 18
    dp: 1 1 1 1 1 1 1 1

    • 当i = 3时,nums[i] = 5,此时在5之前有2和9两个元素,其中2比5小,所以以5结尾的最长递增子序列长度为dp[2] + 1 = 2。

    nums: 10 9 2 5 3 7 101 18
    dp: 1 1 1 2 1 1 1 1

    • 当i = 4时,nums[i] = 3,此时在3之前有2和5两个元素,其中2比3小,所以以3结尾的最长递增子序列长度为dp[2] + 1 = 2。

    nums: 10 9 2 5 3 7 101 18
    dp: 1 1 1 2 2 1 1 1

后面略~~~~~~

完整代码如下:

class Solution {
    public int lengthOfLIS(int[] nums) {
        if (nums == null || nums.length == 0) {
            return 0;
        }
        
        int n = nums.length;
        int[] dp = new int[n];
        dp[0] = 1; 
        
        int result = 1; 
        for (int i = 1; i < n; i++) {
            dp[i] = 1; 
            for (int j = 0; j < i; j++) {
                if (nums[i] > nums[j]) { 
                    dp[i] = Math.max(dp[i], dp[j] + 1); 
                }
            }
            
            result = Math.max(result, dp[i]); 
        }
        
        return result;
    }
}

4 完全平方数

leetcode 279. 完全平方数

动态规划五部曲:

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

**dp[i]:**表示数字i的最少完全平方数的个数。

  1. 确定递推公式

对于数字 i 来说,我们需要遍历所有小于等于 i 的完全平方数 j( j 从 1 到 sqrt(i) ),然后将当前数字 i 减去 j 得到差值,即 i - j 。我们需要找到 dp[ i - j * j ] 的最小值,然后再加上1(表示当前完全平方数 j ),即可得到dp[i]的值。

递推公式为:dp[i] = Math.min(dp[i], dp[i - j * j] + 1),其中 j * j <= i。

  1. 初始化dp数组

Arrays.fill(dp, Integer.MAX_VALUE);

dp[0] = 0;

  1. 确定遍历顺序
// 遍历dp数组
for (int i = 1; i <= n; i++) {
    // 遍历小于等于i的完全平方数j*j
    for (int j = 1; j * j <= i; j++) {
        // 更新dp[i]
        dp[i] = Math.min(dp[i], dp[i - j * j] + 1);
    }
}
  1. 举例推导dp数组

略。。。

完整代码:

class Solution {
    public int numSquares(int n) {
       // 定义dp数组
        int[] dp = new int[n + 1];
        // 初始化dp数组
        Arrays.fill(dp, n+1);
        dp[0] = 0;

        // 遍历dp数组
        for (int i = 1; i <= n; i++) {
            // 遍历小于等于i的完全平方数j*j
            for (int j = 1; j * j <= i; j++) {
                // 更新dp[i]
                dp[i] = Math.min(dp[i], dp[i - j * j] + 1);
            }
        }

        return dp[n]; 
    }
}

当然,这个代码可以再优化一下:(使用Math的api)
减少内层循环的次数:对于小于等于 i 的完全平方数 j ,我们可以通过计算 i - j * j 的平方根得到 j 的最大值,并从最大值开始遍历,这样可以减少内层循环的次数。

class Solution {
    public static int numSquares(int n) {
        // 定义dp数组
        int[] dp = new int[n + 1];
        // 初始化dp数组
        Arrays.fill(dp, n + 1);
        dp[0] = 0;

        // 遍历dp数组
        for (int i = 1; i <= n; i++) {
            // 获取当前数字i的最大完全平方数j*j
            int maxSquare = (int) Math.sqrt(i);
            // 遍历完全平方数j*j
            for (int j = maxSquare; j >= 1; j--) {
                // 更新dp[i]
                dp[i] = Math.min(dp[i], dp[i - j * j] + 1);
            }
        }

        return dp[n];
    }
}

5 跳跃游戏

leetcode 55. 跳跃游戏

动态规划五部曲:

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

dp[i]表示从起点位置到达位置i时能否跳跃到最后一个位置。

  1. 确定递推公式

dp[i] = (dp[j] && nums[j] >= i - j),其中0 <= j < i

  1. 初始化dp数组

初始化dp数组所有位置为false。

  1. 确定遍历顺序

外层循环遍历i从1到n-1,内层循环遍历j从0到i-1。

  1. 举例推导dp数组

以数组nums = [2, 3, 1, 1, 4]为例进行推导:

初始状态:
dp = [false, false, false, false, false]

推导dp[1]:
dp[1] = (dp[0] && nums[0] >= 1 - 0) = (false && 2 >= 1) = false

推导dp[2]:
dp[2] = (dp[0] && nums[0] >= 2 - 0) || (dp[1] && nums[1] >= 2 - 1) = (false && 2 >= 2) || (false && 3 >= 2) = false

推导dp[3]:
dp[3] = (dp[0] && nums[0] >= 3 - 0) || (dp[1] && nums[1] >= 3 - 1) || (dp[2] && nums[2] >= 3 - 2) = (false && 2 >= 3) || (false && 3 >= 3) || (false && 1 >= 3) = false

完整代码如下:

class Solution {
    public boolean canJump(int[] nums) {
        // 获取数组长度
        int n = nums.length;
        // 定义dp数组
        boolean[] dp = new boolean[n];
        // 初始化dp数组
        dp[0] = true;

        // 遍历dp数组
        for (int i = 1; i < n; i++) {
            // 内层循环遍历j
            for (int j = 0; j < i; j++) {
                // 更新dp[i]
                dp[i] = dp[j] && nums[j] >= i - j;
                // 如果dp[i]为true,则跳出内层循环
                if (dp[i]) {
                    break;
                }
            }
        }

        return dp[n - 1];
    }
}

6 解码方法

leetcode 91. 解码方法

动态规划五部曲:

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

dp[i]表示从字符串的起始位置到第i个字符时的解码方法总数。

  1. 确定递推公式

对于dp数组中的每个位置i,我们需要考虑两个情况:

  • 如果第i个字符能够单独解码(即不为0),则dp[i] = dp[i-1],因为第i个字符自身可以作为一个解码方法;
  • 如果第i个字符与前一个字符组成的两位数能够解码(即与前一个字符组成的数字在1到26之间),则dp[i] += dp[i-2],因为组成的两位数可以作为一个解码方法。

则,递推公式为:dp[i] = dp[i-1] + dp[i-2],其中0 <= i < n。

  1. 初始化dp数组

初始化dp数组的长度为n+1,初始值为0。

  1. 确定遍历顺序
for (int i = 1; i <= n; i++) {
    // 如果第i个字符能够单独解码(即不为0)
    if (s.charAt(i - 1) != '0') {
        dp[i] += dp[i - 1];
    }
    // 如果第i个字符与前一个字符组成的两位数能够解码(即与前一个字符组成的数字在1到26之间)
    if (i >= 2 && isValidEncoding(s.substring(i - 2, i))) {
        dp[i] += dp[i - 2];
    }
}
// 判断字符串编码是否在1到26之间
private static boolean isValidEncoding(String s) {
    if (s.charAt(0) == '0') {
        return false;
    }
    int num = Integer.parseInt(s);
    return num >= 1 && num <= 26;
}
  1. 举例推导dp数组

以字符串s = "226"为例进行推导:

初始状态:
dp = [1, 0, 0, 0]

推导dp[1]:
如果第1个字符为2,能够单独解码为"2",所以dp[1] = dp[0] = 1

推导dp[2]:
如果第2个字符为2,能够单独解码为"2",所以dp[2] = dp[1] = 1
如果第1个字符与第2个字符组成的两位数为26,能够解码为"26",所以dp[2] += dp[0],即dp[2] = dp[1] + dp[0] = 1 + 1 = 2

推导dp[3]:
如果第3个字符为6,能够单独解码为"6",所以dp[3] = dp[2] = 2
如果第2个字符与第3个字符组成的两位数为26,能够解码为"26",所以dp[3] += dp[1],即dp[3] = dp[2] + dp[1] = 2 + 1 = 3

最终结果:
dp = [1, 1, 2, 3]

完整代码:

class Solution {
    public static int numDecodings(String s) {
        // 获取字符串的长度
        int n = s.length();
        // 定义dp数组
        int[] dp = new int[n + 1];
        // 初始化dp数组
        dp[0] = 1;

        // 遍历dp数组
        for (int i = 1; i <= n; i++) {
            // 如果第i个字符能够单独解码(即不为0)
            if (s.charAt(i - 1) != '0') {
                dp[i] += dp[i - 1];
            }
            // 如果第i个字符与前一个字符组成的两位数能够解码(即与前一个字符组成的数字在1到26之间)
            if (i >= 2 && isValidEncoding(s.substring(i - 2, i))) {
                dp[i] += dp[i - 2];
            }
        }

        return dp[n];
    }

    // 判断字符串编码是否在1到26之间
    private static boolean isValidEncoding(String s) {
        if (s.charAt(0) == '0') {
            return false;
        }
        int num = Integer.parseInt(s);
        return num >= 1 && num <= 26;
    }
}

不过可以简化一下,就是比较难理解一点,意义一样滴:

class Solution {
    public int numDecodings(String s) {
        int n = s.length();
        int[] f = new int[n + 1];
        f[0] = 1;
        for (int i = 1; i <= n; ++i) {
            if (s.charAt(i - 1) != '0') {
                f[i] += f[i - 1];
            }
            if (i > 1 && s.charAt(i - 2) != '0' 
                && ((s.charAt(i - 2) - '0') * 10 + (s.charAt(i - 1) - '0') <= 26)) {
                f[i] += f[i - 2];
            }
        }
        return f[n];
    }
}

7 不同路径 II

leetcode 63. 不同路径 II

这题就是62的改版,所以复杂了很多,还是建议看代码随想录:动态规划——不同路径

动规五部曲:

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

**dp[i] [j] :**表示从(0 ,0)出发,到(i, j) 有dp[i] [j]条不同的路径。

  1. 确定递推公式

递推公式和62.不同路径一样,dp[i] [j] = dp[i - 1] [j] + dp[i] [j - 1]。

但这里需要注意一点,因为有了障碍,(i, j)如果就是障碍的话应该就保持初始状态(初始状态为0)。

所以代码为:

if (obstacleGrid[i][j] == 0) { // 当(i, j)没有障碍的时候,再推导dp[i][j]
    dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
}
  1. dp数组如何初始化

因为从(0, 0)的位置到(i, 0)的路径只有一条,所以dp[i] [0]一定为1,dp[0] [j]也同理。

但如果(i, 0) 这条边有了障碍之后,障碍之后(包括障碍)都是走不到的位置了,所以障碍之后的dp[i] [0]应该还是初始值0。

如图:

image-20230910161750355

下标(0, j)的初始化情况同理。

所以本题初始化代码为:

int[][] dp = new int[m][n];
for (int i = 0; i < m && obstacleGrid[i][0] == 0; i++) {
    dp[i][0] = 1;
}
for (int j = 0; j < n && obstacleGrid[0][j] == 0; j++) {
    dp[0][j] = 1;
}

注意代码里for循环的终止条件,一旦遇到obstacleGrid[i] [0] == 1的情况就停止dp[i] [0]的赋值1的操作,dp[0] [j]同理

  1. 确定遍历顺序

从递归公式dp[i] [j] = dp [i - 1] [j] + dp[i] [j - 1] 中可以看出,一定是从左到右一层一层遍历,这样保证推导dp[i] [j]的时候,dp[i - 1] [j] 和 dp[i] [j - 1]一定是有数值。

代码如下:

for (int i = 1; i < m; i++) {
    for (int j = 1; j < n; j++) {
        dp[i][j] = (obstacleGrid[i][j] == 0) ? dp[i - 1][j] + dp[i][j - 1] : 0;
    }
}
  1. 举例推导dp数组

image-20230910162011436

完整代码如下:

class Solution {
    public int uniquePathsWithObstacles(int[][] obstacleGrid) {
        int m = obstacleGrid.length;
        int n = obstacleGrid[0].length;
        int[][] dp = new int[m][n];

        //如果在起点或终点出现了障碍,直接返回0
        if (obstacleGrid[m - 1][n - 1] == 1 || obstacleGrid[0][0] == 1) {
            return 0;
        }

        for (int i = 0; i < m && obstacleGrid[i][0] == 0; i++) {
            dp[i][0] = 1;
        }
        for (int j = 0; j < n && obstacleGrid[0][j] == 0; j++) {
            dp[0][j] = 1;
        }

        for (int i = 1; i < m; i++) {
            for (int j = 1; j < n; j++) {
                dp[i][j] = (obstacleGrid[i][j] == 0) ? dp[i - 1][j] + dp[i][j - 1] : 0;
            }
        }
        return dp[m - 1][n - 1];
    }
}

至于118,119我个人觉得并不合适使用动态规划的方式,所以就不写了,over~~

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值