自由的羽毛 发表于 2024-8-14 09:28:34

代码随想录算法训练营day41|动态规划part08

第一题:121. Best Time to Buy and Sell Stock
   贪心法:
class Solution {
    public int maxProfit(int[] prices) {
      // 找到一个最小的购入点
      int low = Integer.MAX_VALUE;
      // res不断更新,直到数组循环完毕
      int res = 0;
      for(int i = 0; i < prices.length; i++){
            low = Math.min(prices, low);
            res = Math.max(prices - low, res);
      }
      return res;
    }
}
   动态规划:版本一
// 解法1
class Solution {
    public int maxProfit(int[] prices) {
      if (prices == null || prices.length == 0) return 0;
      int length = prices.length;
      // dp代表第i天持有股票的最大收益
      // dp代表第i天不持有股票的最大收益
      int[][] dp = new int;
      int result = 0;
      dp = -prices;
      dp = 0;
      for (int i = 1; i < length; i++) {
            dp = Math.max(dp, -prices);
            dp = Math.max(dp + prices, dp);
      }
      return dp;
    }
}
   动态规划:版本二(利用二維數組(和卡哥思路同等),下面還有利用一維滾動數組的更優化版本)
class Solution {
    public int maxProfit(int[] prices) {
      int len = prices.length;
      int dp[][] = new int;
      
      dp = - prices;
      dp = 0;

      for (int i = 1; i < len; i++){
            dp = Math.max(dp[(i - 1) % 2], - prices);
            dp = Math.max(dp[(i - 1) % 2], prices + dp[(i - 1) % 2]);
      }
      return dp[(len - 1) % 2];
    }
}
   动态规划:版本二(利用一維數組)
class Solution {
public int maxProfit(int[] prices) {
    int[] dp = new int;
    // 记录一次交易,一次交易有买入卖出两种状态
    // 0代表持有,1代表卖出
    dp = -prices;
    dp = 0;
    // 可以参考斐波那契问题的优化方式
    // 我们从 i=1 开始遍历数组,一共有 prices.length 天,
    // 所以是 i<=prices.length
    for (int i = 1; i <= prices.length; i++) {
      // 前一天持有;或当天买入
      dp = Math.max(dp, -prices);
      // 如果 dp 被更新,那么 dp 肯定会被更新为正数的 dp
      // 而不是 dp+prices==0 的0,
      // 所以这里使用会改变的dp也是可以的
      // 当然 dp 初始值为 0 ,被更新成 0 也没影响
      // 前一天卖出;或当天卖出, 当天要卖出,得前一天持有才行
      dp = Math.max(dp, dp + prices);
    }
    return dp;
}
} 第二题:122. Best Time to Buy and Sell Stock II
// 动态规划
class Solution
    // 实现1:二维数组存储
    // 可以将每天持有与否的情况分别用 dp 和 dp 来进行存储
    // 时间复杂度:O(n),空间复杂度:O(n)
    public int maxProfit(int[] prices) {
      int n = prices.length;
      int[][] dp = new int;   // 创建二维数组存储状态
      dp = 0;                   // 初始状态
      dp = -prices;
      for (int i = 1; i < n; ++i) {
            dp = Math.max(dp, dp + prices);    // 第 i 天,没有股票
            dp = Math.max(dp, dp - prices);    // 第 i 天,持有股票
      }
      return dp;    // 卖出股票收益高于持有股票收益,因此取
    }
}
//DP using 2*2 Array (下方還有使用一維滾動數組的更優化版本)
class Solution {
    public int maxProfit(int[] prices) {
      int dp[][] = new int ;
      //dp: holding the stock
      //dp: not holding the stock
      dp = - prices;
      dp = 0;

      for(int i = 1; i < prices.length; i++){
            dp = Math.max(dp[(i - 1) % 2], dp[(i - 1) % 2] - prices);
            dp = Math.max(dp[(i - 1) % 2], dp[(i - 1) % 2] + prices);
      }
      return dp[(prices.length - 1) % 2];
    }
}
// 优化空间
class Solution {
    public int maxProfit(int[] prices) {
      int[] dp = new int;
      // 0表示持有,1表示卖出
      dp = -prices;
      dp = 0;
      for(int i = 1; i <= prices.length; i++){
            // 前一天持有; 既然不限制交易次数,那么再次买股票时,要加上之前的收益
            dp = Math.max(dp, dp - prices);
            // 前一天卖出; 或当天卖出,当天卖出,得先持有
            dp = Math.max(dp, dp + prices);
      }
      return dp;
    }
}  
第三题:123. Best Time to Buy and Sell Stock III 
// 版本一
class Solution {
    public int maxProfit(int[] prices) {
      int len = prices.length;
      // 边界判断, 题目中 length >= 1, 所以可省去
      if (prices.length == 0) return 0;

      /*
         * 定义 5 种状态:
         * 0: 没有操作, 1: 第一次买入, 2: 第一次卖出, 3: 第二次买入, 4: 第二次卖出
         */
      int[][] dp = new int;
      dp = -prices;
      // 初始化第二次买入的状态是确保 最后结果是最多两次买卖的最大利润
      dp = -prices;

      for (int i = 1; i < len; i++) {
            dp = Math.max(dp, -prices);
            dp = Math.max(dp, dp + prices);
            dp = Math.max(dp, dp - prices);
            dp = Math.max(dp, dp + prices);
      }

      return dp;
    }
}

// 版本二: 空间优化
class Solution {
    public int maxProfit(int[] prices) {
      int[] dp = new int;
      // 存储两次交易的状态就行了
      // dp代表第一次交易的买入
      dp = -prices;
      // dp代表第一次交易的卖出
      dp = 0;
      // dp代表第二次交易的买入
      dp = -prices;
      // dp代表第二次交易的卖出
      dp = 0;
      for(int i = 1; i <= prices.length; i++){
            // 要么保持不变,要么没有就买,有了就卖
            dp = Math.max(dp, -prices);
            dp = Math.max(dp, dp+prices);
            // 这已经是第二次交易了,所以得加上前一次交易卖出去的收获
            dp = Math.max(dp, dp-prices);
            dp = Math.max(dp, dp+ prices);
      }
      return dp;
    }
}
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页: [1]
查看完整版本: 代码随想录算法训练营day41|动态规划part08