LeetCode——动态规划(Java)

news2025/1/19 20:29:39

动态规划

  • 简介
  • [简单] 509. 斐波那契数
  • [简单] 70. 爬楼梯
  • [简单] 746. 使用最小花费爬楼梯
  • [中等] 62. 不同路径
  • [中等] 63. 不同路径 II
  • [中等] 343. 整数拆分
  • [中等] 96. 不同的二叉搜索树
  • 背包问题
    • 01背包
      • [中等] 416. 分割等和子集
      • [中等] 1049. 最后一块石头的重量 II
      • [中等] 494. 目标和
      • [中等] 474. 一和零
    • 完全背包
      • [中等] 518. 零钱兑换 II
      • [中等] 377. 组合总和 Ⅳ
      • [中等] 322. 零钱兑换
      • [中等] 279. 完全平方数
      • [中等] 139. 单词拆分
  • [中等] 198. 打家劫舍
  • [中等] 213. 打家劫舍 II
  • 树形DP
    • [中等] 337. 打家劫舍 III
  • [简单] 121. 买卖股票的最佳时机
  • [中等] 122. 买卖股票的最佳时机 II
  • [中等] 123. 买卖股票的最佳时机 III
  • [中等] 188. 买卖股票的最佳时机 IV
  • [中等] 309. 买卖股票的最佳时机含冷冻期
  • [中等] 714. 买卖股票的最佳时机含手续费
  • [中等] 300. 最长递增子序列
  • [简单] 674. 最长连续递增序列
  • [中等] 718. 最长重复子数组
  • [中等] 1143. 最长公共子序列

简介

记录一下自己刷题的历程以及代码。写题过程中参考了 代码随想录的刷题路线。会附上一些个人的思路,如果有错误,可以在评论区提醒一下。

[简单] 509. 斐波那契数

原题链接

简单的递归

public int fib(int n) {
    if(n == 0) return 0;
    else if(n == 1) return 1;
    else return fib(n - 1) + fib(n - 2);
}

动态规划

class Solution {
    public int fib(int n) {
        int[] dp = new int[n + 1];
        if(n >= 0) dp[0] = 0;
        if(n >= 1) dp[1] = 1;
        for(int i = 2; i <= n; i++){
            dp[i] = dp[i - 1] + dp[i - 2];
        }
        return dp[n];
    }
}

[简单] 70. 爬楼梯

原题链接

同斐波那契数一样

class Solution {
    public int climbStairs(int n) {
        int[] dp = new int[n + 1];
        if(n >= 1) dp[1] = 1;
        if(n >= 2) dp[2] = 2;
        for(int i = 3; i <= n; i++){
            dp[i] = dp[i - 1] + dp[i - 2];
        }
        return dp[n];
    }
}

[简单] 746. 使用最小花费爬楼梯

原题链接

class Solution {
    public int minCostClimbingStairs(int[] cost) {
        int length = cost.length;
        int[] dp = new int[length + 1];
        dp[0] = 0;
        dp[1] = 0;
        for(int i = 2; i <= length; i++){
            dp[i] = Math.min(dp[i - 1] + cost[i - 1], dp[i - 2] + cost[i - 2]);
        }
        return dp[length];
    }
}

[中等] 62. 不同路径

原题链接

先确定dp数组表示是每个网格有多少路径
机器人只能向下或者向右,这样的话推导式就是dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
第一行和第一列,因为只能选择一个方向,路径都是1,初始化dp时进行赋值。

class Solution {
    public int uniquePaths(int m, int n) {
        int[][] dp = new int[m][n];
        dp[0][0] = 0;
        for(int i = 0; i < m; i++){
            dp[i][0] = 1;
        }
        for(int i = 0; i < n; i++){
            dp[0][i] = 1;
        }
        for(int i = 1; i < m; i++){
            for(int j = 1; j < n; j++){
                dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
            }
        }
        return dp[m - 1][n - 1];
    }
}

[中等] 63. 不同路径 II

原题链接

在上一题的基础中,做障碍判断
初始网格需要做判断,有可能在起点出现障碍而不可达
把第一行和第一列的逻辑改为dp[i][0] = dp[i - 1][0];,因为当出现障碍的时候,第一行或者第一列的网格是不可达的,

class Solution {
    public int uniquePathsWithObstacles(int[][] obstacleGrid) {
        int m = obstacleGrid.length;
        int n = obstacleGrid[0].length;
        int[][] dp = new int[m][n];
        if(obstacleGrid[0][0] == 1) dp[0][0] = 0;
        else dp[0][0] = 1;
        for(int i = 1; i < m; i++){
            if(obstacleGrid[i][0] == 0)
                dp[i][0] = dp[i - 1][0];
        }
        for(int i = 1; i < n; i++){
            if(obstacleGrid[0][i] == 0)
                dp[0][i] = dp[0][i - 1];
        }
        for(int i = 1; i < m; i++){
            for(int j = 1; j < n; j++){
                if(obstacleGrid[i][j] == 1){
                    dp[i][j] = 0;
                    continue;
                }
                dp[i][j] = dp[i - 1][j] + dp[i][j - 1];
            }
        }
        return dp[m - 1][n - 1];
    }
}

[中等] 343. 整数拆分

原题链接

dp数组保存的值是:正整数i拆分后的最大乘积
递推公式就是在循环中 将i拆解为两个数的和,找出不同组合中乘积最大的情况。

初始化的时候注意,dp[2] 以及 dp[3] 用自身值放入数组。

从下图可以知道,下标i == 2 或者 i == 3 的情况下,拆分成两个数以上的乘积值比本身小,意味着之后的数字中如果拆出了3,由于3继续拆分只会比本身更小,所以没有拆分必要。举个例子,手推这个过程的时候,比如10 = 3 + 7,理论上7 = 3 + 4,3与4的乘积大于7,所以最后10 = 3 + 3 + 4,但是3则没有继续拆分的必要。

手推一下这个过程比较容易理解
在这里插入图片描述

注意:如果想保证使用dp数组就能完成所有返回值,可以把 dp[j] 和 j 的大小判断逻辑加入到max的取值中进行判断。但我推导下来,会出现这种情况的其实只有2和3,所以我进行了单独设置。

class Solution {
    public int integerBreak(int n) {
        // 每个正整数可以化成的最大乘积
        int[] dp = new int[n + 1];
        if(n ==2) return 1;
        else if(n == 3) return 2;
        dp[2] = 2;
        if(n >= 3) dp[3] = 3;
        for(int i = 4; i <= n; i++){
            int max = Integer.MIN_VALUE;
            for(int j = i/2; j > 1; j--){
                int num = dp[j] * dp[i - j];
                max = num > max ? num : max;
            }
            dp[i] = max;
        }
        return dp[n];
    }
}

[中等] 96. 不同的二叉搜索树

原题链接

dp数组表示的是 i 个结点有dp[ i ] 种摆放方法摆出二叉搜索树

递推式就是取 j 作为根节点,左边有 j - 1 个结点, 右边 有 i - j 个结点,二者的摆放方法数相乘即为:以j为根节点,节点[1, i] 的摆放方法数。

dp[ 0 ] 取值 为1主要是为了后续相乘时正确处理,也可以理解为, 0 个节点只有一种摆放方式,就是没得摆。

class Solution {
    public int numTrees(int n) {
        int[] dp = new int[n + 1];
        dp[0] = 1;
        dp[1] = 1;
        if(n >= 2) dp[2] = 2;
        for(int i = 3; i <= n; i++) {
            for(int j = 1; j <= i; j++){
                dp[i] += dp[i - j] * dp[j - 1];
            }
        }
        return dp[n];
    }
}

背包问题

01背包

[中等] 416. 分割等和子集

原题链接

动态规划:01背包解法
数组总和 sum / 2 就是背包的最大容量,nums数组看做物品,重量和价值都是nums[ i ],只要最后dp[dp.length - 1] == dp.length - 1就说明能够取到总和一半的组合

可以看一看代码随想录:一维背包中一维背包的设计思想,自己递推的时候脑袋里按照二维背包的思路去规划

dp数组表示:背包限制为i 的 情况下可以取到的最大总和
递推公式:不取当前数字,且限制为i 的最大总和 dp[ j ]
取当前数字,限制为 i 的最大总和dp[j - nums[i]] + nums[i],二者取最大值

二重循环中第一层循环是遍历[0 ,i]的物品允许取的情况,二层循环是背包总和限制为 j 的情况
内层循环需要保证倒序,因为使用一维数组的情况下,正序遍历会把上一层物品状态覆盖。

for(int i = 1; i < nums.length; i++){
            for(int j = dp.length - 1; j >= nums[i]; j--){
                dp[j] = Math.max(dp[j], dp[j - nums[i]] + nums[i]);
            }
        }
class Solution {
    public boolean canPartition(int[] nums) {
        int sum = 0;
        for(int i = 0; i < nums.length; i++)
            sum += nums[i];
        if(sum % 2 == 1) return false;
        int[] dp = new int[sum / 2 + 1];
        for(int i = 1; i < dp.length; i++){
            if(i >= nums[0]) dp[i] = nums[0];
        }
        //倒序遍历是保证每个数字只取一次
        for(int i = 1; i < nums.length; i++){
            for(int j = dp.length - 1; j >= nums[i]; j--){
                dp[j] = Math.max(dp[j], dp[j - nums[i]] + nums[i]);
            }
        }
        if(dp[dp.length - 1] == dp.length - 1) return true;
        return false;
    }
}

[中等] 1049. 最后一块石头的重量 II

原题链接

和上一题 [中等] 416. 分割等和子集 相像,其实就是找出两堆重量尽量相近的石头。

如果给出一个背包,最大容量为j,dp[j] 就是他能取到的最大石头重量和,dp数组最大下标为sum / 2,也就是一个石头堆:dp[dp.length - 1]就是这一半石头堆能够取到的最大重量。sum - dp[dp.length - 1]就是另一个石头堆,且若两个石头堆重量无法相等,后者一定比前者大,所以最后的返回值就是二者之差

class Solution {
    public int lastStoneWeightII(int[] stones) {
        int sum = 0;
        for(int i = 0; i < stones.length; i++)
            sum += stones[i];
        int[] dp = new int[sum / 2 + 1];
        for(int i = 1; i < dp.length; i++){
            if(i >= stones[0]) dp[i] = stones[0];
        }
        //倒序遍历是保证每个数字只取一次
        for(int i = 1; i < stones.length; i++){
            for(int j = dp.length - 1; j >= stones[i]; j--){
                dp[j] = Math.max(dp[j], dp[j - stones[i]] + stones[i]);
            }
        }
        return sum - dp[dp.length - 1] - dp[dp.length - 1];
    }
}

[中等] 494. 目标和

原题链接
在这里插入图片描述

class Solution {
    public int findTargetSumWays(int[] nums, int target) {
        int sum = 0;
        for(int i = 0; i < nums.length; i++){
            sum += nums[i];
        }
        //表示加法集合不是整数,也就是无法取到
        if((sum + target) % 2 != 0) return 0;
        //target 绝对值 大于 sum 同样无法取到
        if((Math.abs(target) > sum)) return 0;
        //表示数字总和取到 j 有dp[j]种方案
        int[] dp = new int[(sum +target) / 2 + 1];
        //target == 0,一个数都不取 为一种方案
        dp[0] = 1;
        //倒序遍历是保证每个数字只取一次
        for(int i = 0; i < nums.length; i++){
            for(int j = dp.length - 1; j >= nums[i]; j--){
                dp[j] += dp[j - nums[i]];
            }
        }
        return dp[dp.length - 1];
    }
}

[中等] 474. 一和零

原题链接

标准的01背包,只是对物品重量的考虑变成二维的。
dp表示:限制 i个0 和 j个1 情况下的最大子集长度
递推公式从二者取最大值:dp[ i ] [ j ] 表示不取当前物品,dp[i - zeroNum][j - oneNum] + 1表示取当前物品
对dp的遍历都是从后往前,因为都是对背包容量的遍历,不能覆盖先前值,把二维遍历对应到普通01背包的一维遍历即可

可以在考虑每个字符串时对dp数组做输出观察状态,更易于理解:

class Solution {
    public int findMaxForm(String[] strs, int m, int n) {
        // 限制 i个0 和 j个1 情况下的最大子集长度
        int[][] dp = new int[m + 1][n + 1];
        for(String str : strs){
            char[] chars = str.toCharArray();
            int zeroNum = 0;
            int oneNum = 0;
            for(char c : chars){
                if(c == '0')zeroNum++;
                else oneNum++;
            }
            for(int i = m; i >= zeroNum; i--){
                for(int j = n; j >= oneNum; j--){
                    dp[i][j] = Math.max(dp[i - zeroNum][j - oneNum] + 1, dp[i][j]);
                }
            }
            System.out.println("当前考虑字符串:" + str);
            for(int i = 0; i <= m; i++){
                for(int j = 0; j <=n ;j++){
                    System.out.print(dp[i][j] + " ");
                }
                System.out.println();
            }
            System.out.println();
        }
        return dp[m][n];
    }
}
public class main {
    public static void main(String[] args) {
        Solution solution = new Solution();
        System.out.println(solution.findMaxForm(new String[]{"10","0001","111001","1","0"}, 5, 3));
    }
}

完全背包

  • 如果求组合数就是外层for循环遍历物品,内层for遍历背包。
    外层遍历物品,考虑过的物品不会再回头考虑,就不用考虑排序问题,每个组合都是唯一的。

  • 如果求排列数就是外层for遍历背包,内层for循环遍历物品。

[中等] 518. 零钱兑换 II

原题链接

突然觉得和回溯法的题目很像,可以看看LeetCode——回溯算法(Java) 中39题,把里面的代码的返回值List<List<Integer>> ans输出size也能通过测试,但是在正式提交时回出现爆内存的情况,因为本题中是不需要具体方案,只需要给出方案数,这或许就是碰到一个题目是使用回溯还是动态规划的一个判断角度。

本题是一个完全背包,跟01背包的差距就是完全背包的物品不限制选取次数

dp数组表示:背包容量为 j 时,最多有dp[ j ] 种方案
递归公式:第二层对背包容量的遍历采用正序,因为区别于01背包,物品可以选择多次。

初始化时:
如果正好选了coins[i]后,也就是j-coins[i] == 0的情况表示这个硬币刚好能选,此时dp[0]为1表示只选coins[i]存在这样的一种选择,可以理解为amount == 0 时选择0个物品为1种方案。

class Solution {
    public int change(int amount, int[] coins) {
        int[]dp = new int[amount + 1];
        dp[0] = 1;
        for (int i = 0; i < coins.length; i++){
            for(int j = coins[i]; j <= amount; j++){
                dp[j] += dp[j - coins[i]];
            }
        }
        return dp[amount];
    }
}

[中等] 377. 组合总和 Ⅳ

同上一题一样,可以和回溯法的题目一起思考,如果本题要把排列都列出来的话,只能使用回溯算法爆搜。本题就是求排列问题,所以外层循环为背包容量。

class Solution {
    public int combinationSum4(int[] nums, int target) {
        int[]dp = new int[target + 1];
        dp[0] = 1;
        for(int i = 1; i <= target; i++){
            for(int j = 0; j < nums.length; j++){
                if(i >= nums[j]){
                    dp[i] += dp[i - nums[j]];
                }
            }
        }
        return dp[target];
    }
}

[中等] 322. 零钱兑换

原题链接

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

递推公式:dp[j]表示不考虑本次循环到的钱币,dp[j - coins[i]] + 1表示考虑一个coins[i]

初始化dp[i] = Integer.MAX_VALUE表示没有方法能够凑到总额j,也用于后续比较最小值被有方法的情况覆盖。

在这里插入图片描述

class Solution {
    public int coinChange(int[] coins, int amount) {
        int[]dp = new int[amount + 1];
        dp[0] = 0;
        for(int i = 1; i < dp.length; i++) dp[i] = Integer.MAX_VALUE;
        for(int i = 0; i < coins.length; i++){
            for(int j = coins[i]; j <= amount; j++){
                if(dp[j - coins[i]] != Integer.MAX_VALUE) {
                    dp[j] = Math.min(dp[j], dp[j - coins[i]] + 1);
                }
            }
        }
        if(dp[amount] == Integer.MAX_VALUE) return -1;
        return dp[amount];
    }
}

[中等] 279. 完全平方数

原题链接

dp数组:和为 j 的完全平方数的最少数量为 dp[j]

递推公式:dp[ j ] 表示不考虑选当前数字 i * i 的取值数量,dp[j - i * i] + 1表示选取一个当前数字i * i 的取值数量,因为是完全背包问题,二层循环从前往后遍历,同一个数字可以多次选取

初始化,除了dp[ 0 ] 用于做最小值比较,并且本身题目 n 范围并不包括0,其余设置Integer.MAX_VALUE,用于被覆盖,因为 1 也属于完全平方数,所以最后的数组每个元素必定是有最小方案数的,不存在还有Integer.MAX_VALUE的情况

class Solution {
    public int numSquares(int n) {
        // 和为 j 的完全平方数的最少数量为 dp[j]
        int[] dp = new int[n + 1];
        dp[0] = 0;
        for(int j = 1; j <= n; j++){
            dp[j] = Integer.MAX_VALUE;
        }
        for(int i = 1; i <= 100; i++){
            for(int j = i * i; j <= n; j++){
                dp[j] = Math.min(dp[j], dp[j - i * i] + 1);
            }
        }
        return dp[n];
    }
}

[中等] 139. 单词拆分

原题链接

只是在基础的完全背包上添加了一些字符串操作

dp数组:从左到右长度为 j 的字符串的是否能够被表示 ,dp[j]为布尔值

递推公式:每一层容量考虑时都需要考虑wordDict中的所有字符串,所以,容量为外层循环,字符串集为内层循环,以 j 为末端的子串如果在wordDict集中,那么它是否能够被表示取决于[j - length]length为子串长度。
并且如果当前层 dp[ j ] 已经为true,找到方案的情况下,就可以跳出本次循环。题目中并没有需要提供方案数。

初始化:dp[0]true用于后续做判断

class Solution {
    public boolean wordBreak(String s, List<String> wordDict) {
        boolean[] dp = new boolean[s.length() + 1];
        dp[0] = true;
        for(int j = 1; j <= s.length() ; j++){
            for(int i = 0; i < wordDict.size(); i++){
                int length = wordDict.get(i).length();
                if(j < length) continue;
                String str = s.substring(j - length, j);
                if(str.equals(wordDict.get(i))){
                    dp[j] = dp[j - length];
                    if(dp[j]) break;
                }
            }
        }
        return dp[s.length()];
    }
}

[中等] 198. 打家劫舍

原题链接

class Solution {
    public int rob(int[] nums) {
        // [0,j] 的 最大收益为 dp[j]
        int[] dp = new int[nums.length];
        dp[0] = nums[0];
        if(nums.length > 1) dp[1] = Math.max(nums[1], dp[0]);
        for(int i = 2; i < nums.length; i++){
            dp[i] = Math.max(dp[i - 1], dp[i - 2] + nums[i]);
        }
        return dp[nums.length - 1];
    }
}

[中等] 213. 打家劫舍 II

原题链接

有点讨巧的方法,调用函数保留线性打家劫舍的方案,调用时两种情况,不考虑头或者不考虑尾,强制把环形问题拆分为线性考虑。

class Solution {
    public int rob(int[] nums) {
        if(nums.length == 1) return nums[0];
        int result1 = rob(nums, 0, nums.length - 1);
        int result2 = rob(nums, 1, nums.length);
        return Math.max(result1, result2);
    } 

    public int rob(int[] nums, int start, int end) {
        int length = end - start;
        // [0,j] 的 最大收益为 dp[j]
        int[] dp = new int[nums.length];
        dp[start] = nums[start];
        if(length > 1) dp[start + 1] = Math.max(nums[start + 1], dp[start]);
        for(int i = start + 2; i < end; i++){
            dp[i] = Math.max(dp[i - 1], dp[i - 2] + nums[i]);
        }
        return dp[end - 1];
    }
}

树形DP

[中等] 337. 打家劫舍 III

原题链接

可以设置二维dp数组保存遍历过的节点状态,但是由于本题只需要子树的状态,所以可以直接使用返回值,int[0] 表示偷窃当前节点, int[1]表示没有盗窃当前节点

class Solution {
    public int rob(TreeNode root) {
        int[] nums = recursion(root);
        return Math.max(nums[0], nums[1]);
    }

    // int[0] 表示偷窃当前节点, int[1]表示没有盗窃当前节点
    public int[] recursion(TreeNode root) {
        if(root == null) return new int[]{0 ,0};
        else if(root.left == null && root.right == null){
            return new int[]{root.val, 0};
        }
        int[] left = recursion(root.left);
        int[] right = recursion(root.right);

        // 偷窃当前节点
        int value0 = root.val + left[1] + right[1];
        // 不偷窃当前节点
        int value1 = Math.max(left[0], left[1]) + Math.max(right[0], right[1]);

        return new int[]{value0, value1};
    }
}

[简单] 121. 买卖股票的最佳时机

原题链接

class Solution {
    public int maxProfit(int[] prices) {
        int[][] dp = new int[prices.length][2];
        //dp[j][0] 表示第j天持有股票的最大现金,dp[j][1]表示第j天不持有股票的最大现金;
        int result = 0;
        dp[0][0] = -prices[0];
        dp[0][1] = 0;
        for(int i = 1; i < prices.length; i++){
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][1] - prices[i]);
            dp[i][1] = Math.max(dp[i - 1][0] + prices[i], dp[i - 1][1]);
        }
        return dp[prices.length - 1][1];
    }
}

[中等] 122. 买卖股票的最佳时机 II

原题链接

class Solution {
    public int maxProfit(int[] prices) {
        int[][] dp = new int[prices.length][2];
        //dp[0] 表示第i - 1不持有股票的最大现金,dp[1]表示第i- 1天持有股票的最大现金;
        dp[0][0] = 0;
        dp[0][1] = -prices[0];
        for(int i = 1; i < prices.length; i++){
            //当天不持有股票 = 昨天不持有股票 || 昨天持有股票 + 今天股票收益
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][1] + prices[i]);
            //当天持有股票 = 昨天持有股票 || 昨天不持有股票 - 今天股票成本
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] - prices[i]);
        }
        return dp[prices.length - 1][0];
    }
}

[中等] 123. 买卖股票的最佳时机 III

原题链接

题目中有五种状态,用dp二维下标 [0, 4]表示,0 未买入,1 第一次买入, 2 第一次卖出, 3 第二次买入,4 第二次卖出。

初始化的时候,其实是代入了当天买入卖出,收益为0 这样的情况。最大的时候一定是卖出的状态,而两次卖出的状态现金最大一定是最后一次卖出。如果第一次卖出已经是最大值了,那么我们可以在当天立刻买入再立刻卖出。所以dp[prices.length - 1][4]已经包含了dp[prices.length - 1][2]的情况。也就是说第二次卖出手里所剩的钱一定是最多的。

其实通过观察动态转移方程可以发现,dp数组是可以压缩为一维的,不过那样写会比较绕,空间上会更节省资源

class Solution {
    public int maxProfit(int[] prices) {
        int[][] dp = new int[prices.length][5];
        // 0 未买入,1 第一次买入, 2 第一次卖出, 3 第二次买入,4 第二次卖出
        dp[0][0] = 0;
        dp[0][1] = -prices[0];
        dp[0][2] = 0;
        dp[0][3] = -prices[0];
        dp[0][4] = 0;
        for(int i = 1; i < prices.length; i++){
            dp[i][0] = dp[i - 1][0];
            dp[i][1] = Math.max(dp[i - 1][0] - prices[i], dp[i - 1][1]);
            dp[i][2] = Math.max(dp[i - 1][1] + prices[i], dp[i - 1][2]);
            dp[i][3] = Math.max(dp[i - 1][2] - prices[i], dp[i - 1][3]);
            dp[i][4] = Math.max(dp[i - 1][3] + prices[i], dp[i - 1][4]);
        }
        return dp[prices.length - 1][4];
    }
}

[中等] 188. 买卖股票的最佳时机 IV

原题链接

在 123. 买卖股票的最佳时机 III 的基础上做扩充就可以了,观察买卖两次的代码发现无非就是对奇数和偶数做区分

class Solution {
    public int maxProfit(int k, int[] prices) {
        int[][] dp = new int[prices.length][2 * k + 1];
        // 0 未买入,1 第一次买入, 2 第一次卖出, 3 第二次买入,4 第二次卖出
        for(int i = 0; i <= 2 * k; i++){
            if(i % 2 != 0) dp[0][i] = -prices[0];
        }
        for(int i = 1; i < prices.length; i++){
            for(int j = 1; j <= 2 * k; j++) {
                if(j % 2 == 0){
                    dp[i][j] = Math.max(dp[i - 1][j - 1] + prices[i], dp[i - 1][j]);
                }else{
                    dp[i][j] = Math.max(dp[i - 1][j - 1] - prices[i], dp[i - 1][j]);
                }
            }
        }
        return dp[prices.length - 1][2 * k];
    }
}

[中等] 309. 买卖股票的最佳时机含冷冻期

原题链接

四种状态:0 未持有股票, 1 买入, 2 卖出, 3表冷冻期

注意0表示的是非冷冻期可操作情况下选择不操作
0:今天不持有股票,昨天也不持有,或者昨天是冷冻期
1:今天买入,昨天不会是卖出,但是需要比较一下是今天买入好还是之前的买入好
2:今天卖出,昨天不能是冷冻期,也不能是卖出
3:今天冷冻期,昨天就是卖出股票

最后得到的最大值,有可能最后一天是没做任何操作,或者最后一天是冷冻期,也有可能最后一天出售的股票,所以需要进行比较

class Solution {
    public int maxProfit(int[] prices) {
        int[][] dp = new int[prices.length][4];
        //0 未持有股票, 1 买入, 2 卖出, 3表冷冻期
        dp[0][0] = 0;
        dp[0][1] = -prices[0];
        dp[0][2] = 0;
        dp[0][3] = 0;
        for(int i = 1; i < prices.length; i++){
            // 今天不持有股票, 昨天也不持有,或者昨天冷冻期(冷冻期另外算)
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][3]);
            // 今天买入,昨天不会是卖出
            int max = Math.max(dp[i - 1][3], dp[i - 1][0]) - prices[i];
            dp[i][1] = Math.max(dp[i - 1][1], max);
            // 今天卖出,昨天不能是冷冻期,也不能是卖出
            dp[i][2] = dp[i - 1][1] + prices[i];
            // 今天冷冻期,昨天卖出
            dp[i][3] = dp[i - 1][2];
        }
        int max = Math.max(dp[prices.length - 1][0], dp[prices.length - 1][2]);
        return Math.max(dp[prices.length - 1][3], max);
    }
}

[中等] 714. 买卖股票的最佳时机含手续费

原题链接

和 [中等] 122. 买卖股票的最佳时机 II没什么区别,卖出的时候把手续费算上即可

class Solution {
    public int maxProfit(int[] prices, int fee) {
        int[][] dp = new int[prices.length][2];
        //dp[0] 表示第i - 1不持有股票的最大现金,dp[1]表示第i- 1天持有股票的最大现金;
        dp[0][0] = 0;
        dp[0][1] = -prices[0];
        for(int i = 1; i < prices.length; i++){
            //当天不持有股票 = 昨天不持有股票 || 昨天持有股票 + 今天股票收益
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][1] + prices[i] - fee);
            //当天持有股票 = 昨天持有股票 || 昨天不持有股票 - 今天股票成本
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] - prices[i]);
        }
        return dp[prices.length - 1][0];
    }
}

[中等] 300. 最长递增子序列

原题链接

dp[i] 表示以 nums[i] 为结尾的最大递增子序列个数

class Solution {
    public int lengthOfLIS(int[] nums) {
        // 以 nums[i] 为结尾的最大递增子序列个数
        int[] dp = new int[nums.length];
        for(int i = 0; i < nums.length; i++)
            dp[i] = 1;
        for(int i = 1; i <nums.length; i++){
            for(int j = 0; j < i; j++){
                if(nums[i] > nums[j])
                    dp[i] = Math.max(dp[i], dp[j] + 1);
            }
        }
        int max = Integer.MIN_VALUE;
        for(int i = 0; i < nums.length; i++){
            max = Math.max(dp[i], max);
        }
        return max;
    }
}

[简单] 674. 最长连续递增序列

原题链接

dp[i] 表示以 nums[i] 为结尾的最大连续递增子序列个数,在遍历nums时,就不需要开二重循环,因为递增序列要求连续,只需要跟前一个数字做比较即可

观察代码会发现,每一次dp[i] = Math.max(dp[i], dp[i - 1] + 1)只用到了前一位的dp状态,可以考虑将数组压缩

class Solution {
    public int findLengthOfLCIS(int[] nums) {
        // 以 nums[i] 为结尾的最大连续递增子序列个数
        int[] dp = new int[nums.length];
        for(int i = 0; i < nums.length; i++)
            dp[i] = 1;
        int max = dp[0];
        for(int i = 1; i <nums.length; i++){
            if(nums[i] > nums[i- 1])
                dp[i] = Math.max(dp[i], dp[i - 1] + 1);
            max = Math.max(dp[i], max);
        }
        return max;
    }
}

[中等] 718. 最长重复子数组

原题链接

dp[i][j] 表示:以 nums1[i] 为结尾,以nums2[j] 为结尾的最大重复子串长度 为 dp[i][j]

dp[i][j] = dp[i - 1][j - 1] + 1;相当于回退去找i 和 j 的前一位,看是否相等,并得到最大重复子串长度,如果nums1[i - 1] != nums2[j - 1]dp[i - 1][j - 1] == 0,dp[i][j]即为1

class Solution {
    public int findLength(int[] nums1, int[] nums2) {
        // 以 nums1[i] 为结尾,以nums2[j] 为结尾的最大重复子串长度 为 dp[i][j]
        int[][] dp = new int[nums1.length][nums2.length];
        int max = Integer.MIN_VALUE;
        for(int i = 0; i < nums1.length; i++){
            if(nums1[i] == nums2[0]) dp[i][0] = 1;
            if(max < dp[i][0]) max = dp[i][0];
        }
        for(int i = 0; i < nums2.length; i++){
            if(nums2[i] == nums1[0]) dp[0][i] = 1;
            if(max < dp[0][i]) max = dp[0][i];
        }

        for(int i = 1; i < nums1.length; i++){
            for(int j = 1; j < nums2.length; j++){
                if(nums1[i] == nums2[j]) dp[i][j] = dp[i - 1][j - 1] + 1;
                if(max < dp[i][j]) max = dp[i][j];
            }
        }
        return max;
    }
}

[中等] 1143. 最长公共子序列

原题链接
dp[i][j] 表示: char1[0, i]char2[0, j] 的最大重复子串长度

class Solution {
    public int longestCommonSubsequence(String text1, String text2) {
        char[] char1 = text1.toCharArray();
        char[] char2 = text2.toCharArray();
        // 以 char1[i] 为结尾,以char2[j] 为结尾的最大重复子串长度 为 dp[i][j]
        int[][] dp = new int[char1.length][char2.length];
        if(char1[0] == char2[0]) dp[0][0] = 1;
        int max = dp[0][0];
        for(int i = 1; i < char1.length; i++){
            if(char1[i] == char2[0]) dp[i][0] = 1;
            else dp[i][0] = dp[i - 1][0];
            if(max < dp[i][0]) max = dp[i][0];
        }
        for(int i = 1; i < char2.length; i++){
            if(char2[i] == char1[0]) dp[0][i] = 1;
            else dp[0][i] = dp[0][i - 1];
            if(max < dp[0][i]) max = dp[0][i];
        }

        for(int i = 1; i < char1.length; i++){
            for(int j = 1; j < char2.length; j++){
                if(char1[i] == char2[j]) {
                    dp[i][j] = dp[i - 1][j - 1] + 1;
                }else{
                    dp[i][j] = Math.max(dp[i - 1][j], dp[i][j - 1]);
                }
                if(max < dp[i][j]) max = dp[i][j];
            }
        }
        return max;
    }
}

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

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

相关文章

redis---哨兵模式Sentinel

上次搭建了一主两从的Redis集群&#xff0c;来实现了一定程度上的高可用。相比一个单节点的Redis来说已经有了很大的提升。 但是这个集群还是有一些问题的&#xff0c;主节点宕机了&#xff0c;我们还是需要手动去把另一台从节点提升为主节点&#xff0c;这样就不能实现真正的…

牛客·没有上司的舞会

题目描述 输入与输出 样例 输入 7 1 1 1 1 1 1 1 1 3 2 3 6 4 7 4 4 5 3 5 0 0 输出 5 题解 思路 dp[i][0] 代表i参加&#xff0c;dp[i][1]代表i不参加按树的dfs序计算最大值找到最大的老板状态转移&#xff1a; u&#xff1a;上司&#xff0c;v&#xff1a;员工 上司去&…

基于springboot的粮仓管理系统

文章目录 项目介绍主要功能截图&#xff1a;部分代码展示设计总结项目获取方式 &#x1f345; 作者主页&#xff1a;超级无敌暴龙战士塔塔开 &#x1f345; 简介&#xff1a;Java领域优质创作者&#x1f3c6;、 简历模板、学习资料、面试题库【关注我&#xff0c;都给你】 &…

腾讯云4核8g服务器承载量?4C8G能支持多少人?

腾讯云4核8G服务器多少钱&#xff1f;腾讯云4核8G轻量应用服务器12M带宽租用价格646元15个月&#xff0c;活动页面 txybk.com/go/txy 活动链接打开如下图所示&#xff1a; 腾讯云4核8G服务器优惠价格 这台4核8G服务器是轻量应用服务器&#xff0c;详细配置为&#xff1a;轻量4核…

spring boot后端controller中接收表单参数校验

校验分为两部分&#xff0c;一部分是前端的输入时就校验&#xff0c;一部分时后端接收参数时的校验。本文提到的是后端接收参数时的校验。这个后端校验的存在有什么意义呢&#xff1f; 比如我们设置前端在输入参数时限制输入不能为空&#xff0c;应该为3-20位非空字符&#xf…

AcWing 731. 毕业旅行问题(每日一题)

原题链接&#xff1a;731. 毕业旅行问题 - AcWing题库 此题难度较大&#xff0c;是2019年字节跳动校招题&#xff0c;里面涉及位运算与状态压缩DP&#xff0c;不会的可以去学习&#xff0c;此题根据个人量力而行。 建议看一下y总的讲解&#xff1a;AcWing 731. 毕业旅行问题&…

IT外包服务:企业数据资产化加速利器

随着数字化时代的兴起&#xff0c;数据成为企业最为重要的资源之一。数据驱动创新对于企业的竞争力和可持续发展至关重要。在这一进程中&#xff0c;IT外包服务发挥着关键作用&#xff0c;加速企业数据资产化进程&#xff0c;为企业提供了重要支持。 首先&#xff0c;IT外包服务…

DXP学习001-原理图的全局编辑

目录 一&#xff0c;元件标注的全局编辑 1&#xff0c;元件的标注 1&#xff09;order of processing排序执行顺序 2&#xff09;※matching options匹配选项 3&#xff09;annotate schematic注释原理图 ① schematic sheet ②annotate scope 注释范围 ③order顺序…

【深度学习】深度学习md笔记总结第3篇:TensorFlow介绍,学习目标【附代码文档】

深度学习笔记完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;深度学习课程&#xff0c;深度学习介绍要求,目标,学习目标,1.1.1 区别,学习目标,学习目标。TensorFlow介绍&#xff0c;2.4 张量学习目标,2.4.1 张量(Tensor),2.4.2 创建张量的指令,2.4.3 张量…

为什么要使用MQ?

我们在学习一个新的技术栈的时候&#xff0c;一定要多思考&#xff0c;为什么要用这个东西&#xff0c;这个东西帮助我们解决了什么问题&#xff0c;他的好处是什么&#xff0c;这样有利于我们加深对这个东西的理解。 下面开始今天的正文&#xff0c;我们为什么要使用Mq呢&…

Three.js——scene场景、几何体位置旋转缩放、正射投影相机、透视投影相机

个人简介 &#x1f440;个人主页&#xff1a; 前端杂货铺 &#x1f64b;‍♂️学习方向&#xff1a; 主攻前端方向&#xff0c;正逐渐往全干发展 &#x1f4c3;个人状态&#xff1a; 研发工程师&#xff0c;现效力于中国工业软件事业 &#x1f680;人生格言&#xff1a; 积跬步…

leetcode代码记录(打家劫舍 III

目录 1. 题目&#xff1a;2. 我的代码&#xff1a;小结&#xff1a; 1. 题目&#xff1a; 小偷又发现了一个新的可行窃的地区。这个地区只有一个入口&#xff0c;我们称之为 root 。 除了 root 之外&#xff0c;每栋房子有且只有一个“父“房子与之相连。一番侦察之后&#xf…

红黑树介绍与模拟实现(insert+颜色调整精美图示超详解哦)

红黑树 引言红黑树的介绍实现结点类insert搜索插入位置插入调整当parent为gparent的左子结点当parent为gparent的右子结点 参考源码测试红黑树是否合格总结 引言 在上一篇文章中我们认识了高度平衡的平衡二叉树AVL树&#xff1a;戳我看AVL树详解哦 &#xff08;关于旋转调整的…

Sealos 一键部署FastGPT的解决方案

&#x1f9d9;‍♂️ 诸位好&#xff0c;吾乃斜杠君&#xff0c;编程界之翘楚&#xff0c;代码之大师。算法如流水&#xff0c;逻辑如棋局。 &#x1f4dc; 吾之笔记&#xff0c;内含诸般技术之秘诀。吾欲以此笔记&#xff0c;传授编程之道&#xff0c;助汝解技术难题。 &#…

1.数据结构和算法

文章目录 数据结构逻辑结构集合结构线性结构树形结构图形结构 物理结构顺序存储结构链式存储结构 算法基本特性目标 总结数据结构总结算法总结 数据结构 「数据结构」指的是&#xff1a;数据的组织结构&#xff0c;用来组织、存储数据。 逻辑结构 逻辑结构&#xff08;Logic…

解决oracle数据库乱码

解决oracle数据库乱码 [oraclep19cstd dbca]$ vim ~/.bash_profile #再文件末尾加上 export NLS_LANGAMERICAN_AMERICA.AL32UTF8[oraclep19cstd dbca]$ source ~/.bash_profile [oraclep19cstd dbca]$ sqlplus / as sysdba

Spring声明式事务以及事务传播行为

Spring声明式事务以及事务传播行为 Spring声明式事务1.编程式事务2.使用AOP改造编程式事务3.Spring声明式事务 事务传播行为 如果对数据库事务不太熟悉&#xff0c;可以阅读上一篇博客简单回顾一下&#xff1a;MySQL事务以及并发访问隔离级别 Spring声明式事务 事务一般添加到…

安装Docker(CentOS)

Docker 分为 CE 和 EE 两大版本。CE 即社区版&#xff08;免费&#xff0c;支持周期 7 个月&#xff09;&#xff0c;EE 即企业版&#xff0c;强调安全&#xff0c;付费使用&#xff0c;支持周期 24 个月。 Docker CE 分为 stable test 和 nightly 三个更新频道。 官方网站上…

212 基于matlab的双稳态随机共振的算法

基于matlab的双稳态随机共振的算法&#xff0c;分析信噪比随系统参数a,b及乘性噪声和加性噪声的增益变化曲线&#xff0c;60个数据样本可供选择。程序已调通&#xff0c;可直接运行。 212 双稳态随机共振 信噪比增益变化曲线 - 小红书 (xiaohongshu.com)

【SQL Server的详细使用教程】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…