Bootstrap

动态规划之股票问题123

股票问题作为动态规划里十分经典的题目,也有着一些难度,在这里,我试图给出一套股票问题的模板。

使用这套模板,打到资本主义的压榨。

121. 买卖股票的最佳时机
122. 买卖股票的最佳时机 II
123. 买卖股票的最佳时机 III
188. 买卖股票的最佳时机 IV
309. 最佳买卖股票时机含冷冻期
714. 买卖股票的最佳时机含手续费

问题描述:

给定一个数组,它的第 i 个元素是一支给定的股票在第 i 天的价格。

设计一个算法来计算你所能获取的最大利润。你最多可以完成 两笔 交易。

注意: 你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/best-time-to-buy-and-sell-stock-iii
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

 示例:

输入: [3,3,5,0,0,3,1,4]
输出: 6
解释: 在第 4 天(股票价格 = 0)的时候买入,在第 6 天(股票价格 = 3)的时候卖出,这笔交易所能获得利润 = 3-0 = 3 。
     随后,在第 7 天(股票价格 = 1)的时候买入,在第 8 天 (股票价格 = 4)的时候卖出,这笔交易所能获得利润 = 4-1 = 3 

分析:

又加了一个限制,只能购买两次,这已然难不倒我们,我们只需要把第 i 天,第 j 次交易完成后的收益记录下来了。

设dp0[m][n],为第 i 天, 第 j 次交易,且手中没有没有股票(卖出)

设dp1[m][n],为di i 天,第 j 次交易,且手中有股票(买入)

这里第 i 天卖出股票的的状态不仅和 第 i - 1天的买入有关系,还和 j - 1 次交易有关系。

第 i 天买入的状态只和上一次交易完成时的状态有关,上一次交易完成了,就不会干扰到新的交易。

得出dp方程

  • dp0[i][j] = max(dp0[i][j], dp1[i-1][j-1] + price[i])
  • dp1[i][j] = max(dp1[i][j], dp0[i-1][j] - price[i])

上代码:

class Solution:
    def maxProfit(self, prices: List[int]) -> int:
        # dp[i][j]  i 天,j次
        # dp0  手头没有股票
        # dp1  手头有股票
        dp0 = [[0]*3 for _ in range(len(prices))]
        dp1 = [[0]*3 for _ in range(len(prices))]
        
        for j in range(3):
            dp1[0][i] = -prices[0]
        for i in range(1, len(prices)):
            dp1[i][0] = max(dp1[i-1][0], -prices[i])
            for j in range(1,3):
                dp0[i][j] = max(dp0[i-1][j], dp1[i-1][j-1] + prices[i])
                dp1[i][j] = max(dp1[i-1][j], dp0[i-1][j] - prices[i])
        for i in range(len(dp0)):
            print(dp0[i])
        return dp0[-1][-1]

说明:一次交易都没完成的时候,也就是需要买入股票,所以需要初始化dp1[i][0] = max(dp1[i-1][0], -prices[i])

 

 

 

 

;