[Hdp] lc123. 买卖股票的最佳时机 III(dp+前后缀分解)

1. 题目来源

链接:lc123. 买卖股票的最佳时机 III

2. 题目说明

在这里插入图片描述

3. 题目解析

该股票问题:最多进行 2 笔交易,每笔交易不能重叠(指买入下支股票前需要将手中股票卖出)

股票问题四部曲:
[Edp] lc121. 买卖股票的最佳时机(dp)
[Edp] lc122. 买卖股票的最佳时机 II(dp+思维)
[Hdp] lc123. 买卖股票的最佳时机 III(dp+前后缀分解)
[Hdp] lc188. 买卖股票的最佳时机 IV(状态机模型dp+线性dp+空间优化)

采用两种思路:前后缀分解(专用于处理 2 次交易,或者和 2 相关的问题,是一个很常用的思路)、dp(dp 在这大材小用,dp 可以直接求出 k 次交易的最大收益情况)

方法一:前后缀分解

思路:

  • 枚举的时候可以枚举两次交易的分界点,不妨将分界点取为第二次交易的买入时间,记为第 i
  • 那么第一次交易就在 1~i 天,第二次交易就在第 i 天之后
  • 问题转化为求解在这两个前后缀中各买入一次股票的最大值
  • 这两个时间段是独立的,那么我们可以采用 [Edp] lc121. 买卖股票的最佳时机(dp) 的思想。在求解后缀最大价值的时候,可以反向扫描,保存 i+1 ~ n 的最大值,记为 max,这样利用最大值减去当前值就是能获得的最大价值。
  • 还要预处理得到 f(i-1) 的值,代表 1 ~ i-1 段操作一次的最大收益,这个完全就和 [Edp] lc121. 买卖股票的最佳时机(dp) 一样。f(i) = max(pi - minp, f(i-1))
  • 那么总的收益就是 f(i-1) + max - i

经过前后缀分解,我们就将问题分解为两个完全独立的子问题。分别求取前后缀中的最值。可以通过预处理前后缀,然后枚举分界点即可。

总结: res += max{0,第 i 天的价格 - 前 i - 1 天的价格}

代码:

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        int n = prices.size();
        vector<int> f(n + 2);                           // 枚举前缀,且预处理前缀最大收益,下标从1开始
        for (int i = 1, minp = INT_MAX; i <= n; ++i) {  // minp存前段最小值
            f[i] = max(f[i - 1], prices[i - 1] - minp); // f[i]有两种转移方式
            minp = min(minp, prices[i - 1]);            // 更新minp
        }

        int res = 0;
        for (int i = n, maxp = 0; i; --i) {             // 枚举分界点,动态计算后缀最大收益
            res = max(res, maxp - prices[i - 1] + f[i - 1]); // maxp存后端最大值
            maxp = max(maxp, prices[i - 1]);            // 更新maxp
        }
        return res;
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ypuyu

如果帮助到你,可以请作者喝水~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值