Files
leetcode-master/problems/0123.买卖股票的最佳时机III.md
youngyangyang04 2d739ff7e1 Update
2020-12-29 09:31:49 +08:00

4.4 KiB
Raw Blame History

思路

这道题目相对 121.买卖股票的最佳时机 和 122.买卖股票的最佳时机II 难了不少。

关键在于至多买卖两次,这意味着可以买卖一次,可以买卖两次,也可以不买卖。

接来下我用动态规划五部曲详细分析一下:

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

一天一共就有五个状态, 0. 没有操作

  1. 第一次买入
  2. 第一次卖出
  3. 第二次买入
  4. 第二次卖出

dp[i][j]中 i表示第i天j为 [0 - 4] 五个状态dp[i][j]表示第i天状态j所剩最大现金。

确定递推公式

dp[i][0] = dp[i - 1][0];

需要注意dp[i][1]表示的是第i天买入股票的状态并不是说一定要第i天买入股票这是很多同学容易陷入的误区

达到dp[i][1]状态,有两个具体操作:

  • 操作一第i天买入股票了那么dp[i][1] = dp[i-1][0] - prices[i]
  • 操作二第i天没有操作而是沿用前一天买入的状态dp[i][1] = dp[i - 1][0]

那么dp[i][1]究竟选 dp[i-1][0] - prices[i]还是dp[i - 1][0]呢?

一定是选最大的,所以 dp[i][1] = max(dp[i-1][0] - prices[i], dp[i - 1][0]);

同理dp[i][2]也有两个操作:

  • 操作一第i天卖出股票了那么dp[i][2] = dp[i - 1][i] + prices[i]
  • 操作二第i天没有操作沿用前一天卖出股票的状态dp[i][2] = dp[i - 1][2]

所以dp[i][2] = max(dp[i - 1][i] + prices[i], dp[i][2])

同理可推出剩下状态部分:

dp[i][3] = max(dp[i - 1][3], dp[i - 1][2] - prices[i]); dp[i][4] = max(dp[i - 1][4], dp[i - 1][3] + prices[i]);

dp数组如何初始化

第0天没有操作这个最容易想到就是0dp[0][0] = 0;

第0天做第一次买入的操作dp[0][1] = -prices[0];

第0天做第一次卖出的操作这个初始值应该是多少呢

首先卖出的操作一定是收获利润整个股票买卖最差情况也就是没有盈利即全程无操作现金为0

从递推公式中可以看出每次是取最大值那么既然是收获利润如果比0还小了就没有必要收获这个利润了。

所以dp[0][2] = 0;

第0天第二次买入操作初始值应该是多少呢

不用管第几次,现在手头上没有现金,只要买入,现金就做相应的减少。

所以第二次买入操作初始化为dp[0][3] = -prices[0];

同理第二次卖出初始化dp[0][4] = 0;

确定遍历顺序

从递归公式其实已经可以看出一定是从前向后遍历因为dp[i]依靠dp[i - 1]的数值。

举例推导dp数组

以输入[1,2,3,4,5]为例

123.买卖股票的最佳时机III

红色为最终求解。

因为利润最大一定是卖出的状态所以最终最大利润是max(dp[4][2], dp[4][4]);

C++代码

以上五步都分析完了,不难写出如下代码:

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        if (prices.size() == 0) return 0;
        vector<vector<int>> dp(prices.size(), vector<int>(5, 0));
        dp[0][0] = 0;
        dp[0][1] = -prices[0];
        dp[0][3] = -prices[0];
        for (int i = 1; i < prices.size(); i++) {
            dp[i][0] = dp[i - 1][0];
            dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] - prices[i]);
            dp[i][2] = max(dp[i - 1][2], dp[i - 1][1] + prices[i]);
            dp[i][3] = max(dp[i - 1][3], dp[i - 1][2] - prices[i]);
            dp[i][4] = max(dp[i - 1][4], dp[i - 1][3] + prices[i]);
        }
        return max(dp[prices.size() - 1][2], dp[prices.size() - 1][4]);

    }
};
  • 时间复杂度O(n)
  • 空间复杂度O(n * 5)

当然,大家在网上看到的题解还有一种优化空间写法,如下:

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        if (prices.size() == 0) return 0;
        vector<int> dp(5, 0);
        dp[0] = 0;
        dp[1] = -prices[0];
        dp[3] = -prices[0];
        for (int i = 1; i < prices.size(); i++) {
            dp[1] = max(dp[1], dp[0] - prices[i]);
            dp[2] = max(dp[2], dp[1] + prices[i]);
            dp[3] = max(dp[3], dp[2] - prices[i]);
            dp[4] = max(dp[4], dp[3] + prices[i]);
        }
        return max(dp[2], dp[4]);

    }
};

但这种写法dp[2] 利用的是当天的dp[1],我还没有理解为什么这种写法也可以通过,网上的题解也没有做出解释,可能这就是神代码吧,欢迎大家来讨论一波!