代码随想录算法训练营Day-41动态规划08 | 121. 买卖股票的最佳时机、122.买卖股票的最佳时机II、123.买卖股票的最佳时机III

121. 买卖股票的最佳时机

**1.dp数组含义:**dpi0、dpi1分别代表第i天,不持有股票和持有股票获得的最大价值;

2.递推公式:

不持有股票:dpi0 = max(dpi-10, dpi-11+pricesi)

两种情况,1. 第i-1天就没有持有股票,第i天不操作,所以延续不持有的价值;2. 第i-1天持有股票了,到第i天卖出,导致第i天不持有,所以是 dpi-11+pricesi。两种情况取最大值。

持有股票:dpi1 = max(dpi-11, -pricesi)

两种情况,1. 第i-1天就持有股票,第i天不操作,所以延续持有的价值;2. 第i-1天不持有股票,到第i天买入,导致第i天持有,所以是 -pricesi。两种情况取最大值。

**3.初始化:**只需初始化第0天持有或不持有手中的价值,分别是-prices0和0

**4.遍历顺序:**顺序遍历,从1到尾索引;

注:由于递推只用到了dpi-1,所以可以只创建大小为2*2的数组,滚动更新第i-1天和第i天即可,通过取模实现。

cpp 复制代码
class Solution {
public:
    int maxProfit(vector<int>& prices) {
        vector<vector<int>> dp(2, vector<int>(2));

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

        for(int i=1;i<prices.size();i++){
            dp[i%2][0] = max(dp[(i-1)%2][0], dp[(i-1)%2][1]+prices[i]);
            dp[i%2][1] = max(dp[(i-1)%2][1], -prices[i]);
        }
        return dp[(prices.size()-1) %2][0];
    }
};

122.买卖股票的最佳时机II

和上一题的区别在于:本题可以多次买卖,因此对于第i天持有股票了来说,不一定是第几次持有,所以买入第i天股票时手里的钱不确定是不是0,所以持有股票的递推公式需要改为

cpp 复制代码
dp[i][1] = max(dp[i-1][1], -prices[i]);

应改为:

cpp 复制代码
dp[i][1] = max(dp[i-1][1], dp[i-1][0]-prices[i]);
cpp 复制代码
class Solution {
public:
    int maxProfit(vector<int>& prices) {
        vector<vector<int>> dp(2, vector<int>(2));

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

        for(int i=1;i<prices.size();i++){
            dp[i%2][0] = max(dp[(i-1)%2][0], dp[(i-1)%2][1]+prices[i]);
            dp[i%2][1] = max(dp[(i-1)%2][1], dp[(i-1)%2][0]-prices[i]);
        }
        return dp[(prices.size()-1) %2][0];
    }
};

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

和前两题区别在于,本题最多只能买卖两次,所以全部覆盖状态比较多,需要分成dpi0、dpi1、dpi2、dpi3、dpi4,分别代表第i天不操作、第一次持有、第一次不持有、第二次持有、第二次不持有;

**递推公式:**dpi0延续即可;dpi1由第i-1天的1延续而来,或者0买入;dpi2由第i-1天的2延续而来,或者1卖出;dpi3由第i-1天的3延续而来,或者2买入;dpi4由第i-1天的4延续而来,或者3卖出;

**初始化:**只需要考虑第0天的五种状态,只有持有的时候才会不是0,呈现出负的第一天股票价格,其余都相当于同一天多次买卖,不持有股票,即为0;

cpp 复制代码
class Solution {
public:
    int maxProfit(vector<int>& prices) {
        vector<vector<int>> dp(2, vector<int>(5,0));

        dp[0][1] = -prices[0];//第一次持有
        dp[0][3] = -prices[0];//第二次持有

        for(int i=1;i<prices.size();i++){
            dp[i%2][0] = dp[(i-1)%2][0];
            dp[i%2][1] = max(dp[(i-1)%2][1], dp[(i-1)%2][0]-prices[i]);
            dp[i%2][2] = max(dp[(i-1)%2][2], dp[(i-1)%2][1]+prices[i]);
            dp[i%2][3] = max(dp[(i-1)%2][3], dp[(i-1)%2][2]-prices[i]);
            dp[i%2][4] = max(dp[(i-1)%2][4], dp[(i-1)%2][3]+prices[i]);
        }
        return dp[(prices.size()-1) %2][4];
    }
};

相关推荐
JieE21218 小时前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack201 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树1 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE2122 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2122 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术2 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
clint4562 天前
C++进阶(1)——前景提要
c++
用户497863050732 天前
(一)小红的数组操作
算法·编程语言