第五十一天| 309.最佳买卖股票时机含冷冻期、714.买卖股票的最佳时机含手续费

文章讲述了使用动态规划解决股票买卖问题,包括122.买卖股票的最佳时机II和LeetCode中的股票交易问题,涉及冷冻期限制和手续费计算,通过状态转移和dp数组设计递推公式。

第四十八天| 121. 买卖股票的最佳时机、122.买卖股票的最佳时机II-优快云博客

第五十天| 123.买卖股票的最佳时机III、188.买卖股票的最佳时机IV-优快云博客

Leetcode 309.最佳买卖股票时机含冷冻期 

题目链接:309 最佳买卖股票时机含冷冻期

题干:给定一个整数数组prices,其中第  prices[i] 表示第 i 天的股票价格 。​

设计一个算法计算出最大利润。在满足以下约束条件下,你可以尽可能地完成更多的交易(多次买卖一支股票):

  • 卖出股票后,你无法在第二天买入股票 (即冷冻期为 1 天)。

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

思考:动态规划。本题是在 122.买卖股票的最佳时机II 的基础上加上冷冻期。

  • 确定dp数组以及下标的含义(本题的难点)

dp[i][j]:第i天,状态为j,所剩的最多现金为dp[i][j]。

出现冷冻期后,状态比较复杂度。尤其要注意 不进行买入卖出 的几种情况。

具体可以区分出如下四个状态:

  • 状态一:持有股票状态(今天 或 之前就买入了股票然后没有操作,一直持有)
  • 不持有股票状态,这里就有两种卖出股票状态
    • 状态二:保持卖出股票状态(前一天处于冷冻期 或 之前就是卖出股票状态,一直没操作)
    • 状态三:今天卖出股票(真实卖出股票)
  • 状态四:今天为冷冻期状态,但冷冻期状态不可持续,只有一天!

 以上状态j分别用0,1,2,3代指。

本题之所以要将不持有股票的状态分为状态二和状态三,是由于如果本题的冷冻期从不持有股票的状态推出,则冷冻期连续,不止一天,与题干矛盾。

  • 确定递推公式

达到买入股票状态(状态一)即:dp[i][0],有两个具体操作:

  • 操作一:前一天就是持有股票状态(状态一),dp[i][0] = dp[i - 1][0]
  • 操作二:今天买入股票,有两种情况
    • 前一天是冷冻期(状态四),dp[i - 1][3] - prices[i]
    • 前一天是保持卖出股票的状态(状态二),dp[i - 1][1] - prices[i]

那么dp[i][0] = max(dp[i - 1][0], dp[i - 1][3] - prices[i], dp[i - 1][1] - prices[i]);


达到保持卖出股票状态(状态二)即:dp[i][1],有两个具体操作:

  • 操作一:前一天就是状态二,dp[i - 1][1]
  • 操作二:前一天是冷冻期(状态四),dp[i - 1][3]

dp[i][1] = max(dp[i - 1][1], dp[i - 1][3]);


达到今天就卖出股票状态(状态三),即:dp[i][2] ,只有一个操作:

昨天一定是持有股票状态(状态一),今天卖出。即:dp[i][2] = dp[i - 1][0] + prices[i];


达到冷冻期状态(状态四),即:dp[i][3],只有一个操作:

昨天卖出了股票(状态三)。dp[i][3] = dp[i - 1][2];

  • dp数组如何初始化

从递推公式可以看出,基础为dp[0][j],即第0天如何初始化。

如果是持有股票状态(状态一)那么:dp[0][0] = -prices[0],一定是当天买入股票。

之后三种状态的初始化,从定义难以解释,因此我们直接从递推公式来推导。

如果i为1,第1天买入股票,那么递归公式中需要计算 dp[i - 1][1] - prices[i] ,即 dp[0][1] - prices[1],此处的 dp[0][1] (即第0天的保持卖出股票状态(状态二))相当于第0天的利润作本金,因此初始为0。

同理第0天的今天卖出股票(状态三),dp[0][2]也应初始化为0。

此外由于冷冻期不操作,状态四和状态三的值恒等。因此dp[0][3]也初始为0。

  • 确定遍历顺序

从递归公式上可以看出,dp[i] 依赖于 dp[i-1],所以是从前向后遍历。

  • 举例推导dp数组

以 [1,2,3,0,2] 为例,dp数组如下:

最后 获取最大金额一定不是持有股票的状态,只可能是最后一天卖出股票(状态三)或最后一天不操作(状态二和状态四),并且返回这三种状态的最大值。 

代码:

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        vector<vector<int>> dp(prices.size(), vector<int>(4));
        dp[0][0] = - prices[0];
        dp[0][1] = 0;
        dp[0][2] = 0;
        dp[0][3] = 0;

        for (int i = 1; i < prices.size(); i++) {
            //保持持有股票、之前处于卖出股票本次买入股票、之前处于冷冻期本次买入股票
            dp[i][0] = max(dp[i - 1][0], max(dp[i - 1][1], dp[i - 1][3]) - prices[i]);
            //保持卖出股票状态、之前处于冷冻期状态本次处于卖出股票状态
            dp[i][1] = max(dp[i - 1][1], dp[i - 1][3]);
            //之前处于持有股票状态本次真实卖出股票
            dp[i][2] = dp[i - 1][0] + prices[i];
            //之前真实卖出股票本次处于冷冻期
            dp[i][3] = dp[i - 1][2];
        }

        //最后一天卖出股票 或 最后一天无操作
        return max(dp[prices.size() - 1][2], max(dp[prices.size() - 1][1], dp[prices.size() - 1][3]));
    }
};

Leetcode 714.买卖股票的最佳时机含手续费

题目链接:714 买卖股票的最佳时机含手续费

题干:给定一个整数数组 prices,其中 prices[i]表示第 i 天的股票价格 ;整数 fee 代表了交易股票的手续费用。

你可以无限次地完成交易,但是你每笔交易都需要付手续费。如果你已经购买了一个股票,在卖出它之前你就不能再继续购买股票了。

返回获得利润的最大值。

注意:这里的一笔交易指买入持有并卖出股票的整个过程,每笔交易你只需要为支付一次手续费。

思考:动态规划。本题与 122.买卖股票的最佳时机II 的区别就是这里需要多一个减去手续费的操作。减去手续费的操作既可以放在购入股票时也可以放在卖出股票时。(下面的代码将费用放在购票时)。递推公式和初始化稍加修改即可。

代码:

class Solution {
public:
    int maxProfit(vector<int>& prices, int fee) {
        //dp[i][0]:第i天持有股票状态    dp[i][1]:第i天不持有股票状态
        vector<vector<int>> dp(prices.size(), vector<int>(2));
        dp[0][0] = - prices[0] - fee;       //买入股票时考虑手续费
        dp[0][1] = 0;

        for (int i = 1; i < prices.size(); i++) {
            dp[i][0] = max(dp[i - 1][0], dp[i - 1][1] - prices[i] - fee);   //一直持有股票 或 第i天买入股票(上一次交易利润作本金)
            dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] + prices[i]);         //第i天卖出股票 或 之前就卖出股票
        }
        return dp[prices.size() - 1][1]; 
    }
};

股票问题总结:

股票问题的重点有两处:

        对于股票交易次数的限制 可以从 每次交易的本金 和 交易状态考虑。只需一次交易则交易本金一直为0;无限次交易则将上次交易的利润当作本次交易的本金;有限次(大于一次)交易通过添加 不同次交易状态 来同步更新 保证不超过指定交易次数。

​​​​​​​        对于交易情况的限制 可以从 交易状态考虑。本质上和有限次交易次数限制一样,但是要考虑具体的状态种类 来确定dp数组含义 以及 考虑全各种状态之间的转换情况 来确定递推公式。

<think>好的,我需要帮助用户理解冷冻股票买卖最佳时机的动态规划算法,并给出Python实现。首先,我得回顾一下用户提供的引用资料,特别是引用[5]中的代码和思路。用户的问题是关于动态规划解决这个问题的具体步骤和代码实现。 首先,这个问题的主要约束是卖出股票后有一冷冻,不能立即买入。动态规划通常适用于这类问题,因为它可以分解问题到每的状态,并记录每个状态下的最大收益。根据引用[5],状态被分为四种:保持买入状态、保持卖出状态、当卖出、冷冻。我需要理清每个状态之间的转移关系。 保持买入状态(状态0):可能是前一已经是买入状态,或者当买入。而当买入又可能来自前一冷冻或保持卖出状态。这里需要注意,买入时必须是前一没有卖出操作,即不在冷冻。 保持卖出状态(状态1):可能是前一已经是卖出状态,或者前一冷冻。这表示当没有进行任何操作,处于可买入但未买入的状态。 当卖出(状态2):这只能在当卖出,所以前一的必须处于买入状态,卖出后进入冷冻冷冻(状态3):只能在前一卖出后进入,所以当的状态由前一的卖出状态转移而来。 接下来,初始化部分,第一只能买入或者不操作,所以状态0的初始值为-prices[0],其他状态因为没有卖出,收益为0。 然后,遍历每一的价格,根据状态转移方程更新每个状态的最大收益。最后,取最后一所有状态的最大值,因为最终可能处于卖出或冷冻状态。 需要检查代码中的转移方程是否正确,比如状态0的转移是否考虑了所有可能的前驱状态。例如,在状态0的更新中,是否允许从冷冻(状态3)或保持卖出(状态1)后买入?是的,因为冷冻之后的下一可以买入,而保持卖出状态时也可以随时买入,只要没有卖出操作在前一。 用户可能需要更详细的解释,比如每个状态的定义和转移逻辑,以及为什么这样设计状态。此外,代码中的初始化是否正确,循环中的处理是否覆盖了所有情况。 最后,需要确保给出的Python代码正确无误,并且与引用[5]中的代码一致,同时解释清楚每一步的作用。可能还需要讨论时间复杂度和空间复杂度,以及优化方法,比如状态压缩,减少空间使用。</think>### 买卖股票最佳时机冷冻的动态规划解析 #### 问题核心 在股票交易中,卖出后需等待一冷冻才能再次买入,需通过动态规划合理规划买卖操作以实现最大收益。 --- ### 一、动态规划状态定义 定义四个状态描述每日可能的交易状态: 1. **状态0**:保持买入状态(当日持有股票) 2. **状态1**:保持卖出状态(当日未持有股票,且未处于冷冻) 3. **状态2**:当日卖出股票 4. **状态3**:处于冷冻(仅持续一) --- ### 二、状态转移方程 1. **状态0**:$dp[i][0] = \max(dp[i-1][0], \max(dp[i-1][1] - p, dp[i-1][3] - p))$ - 延续前一的买入状态,或从非冷冻(状态1/3)买入 2. **状态1**:$dp[i][1] = \max(dp[i-1][1], dp[i-1][3])$ - 延续前一的卖出状态,或前一冷冻 3. **状态2**:$dp[i][2] = dp[i-1][0] + p$ - 必须在前一持有股票时卖出 4. **状态3**:$dp[i][3] = dp[i-1][2]$ - 仅在前一卖出后进入冷冻 --- ### 三、Python代码实现 ```python def maxProfit(prices): n = len(prices) if n == 0: return 0 dp = [[0] * 4 for _ in range(n)] dp[0][0] = -prices[0] # 初始买入状态 for i in range(1, n): # 状态0: 保持买入或当买入(需从前一的卖出/冷冻转移) dp[i][0] = max(dp[i-1][0], max(dp[i-1][1] - prices[i], dp[i-1][3] - prices[i])) # 状态1: 保持卖出或从冷冻转移 dp[i][1] = max(dp[i-1][1], dp[i-1][3]) # 状态2: 当卖出(必须前一持有) dp[i][2] = dp[i-1][0] + prices[i] # 状态3: 冷冻(前一卖出) dp[i][3] = dp[i-1][2] return max(dp[-1]) # 最终取所有状态的最大值 ``` --- ### 四、算法分析 1. **时间复杂度**:$O(n)$,遍历一次价格数组 2. **空间复杂度**:$O(n)$,可优化为$O(1)$(用变量代替二维数组)[^5] 3. **关键点**:明确状态转移关系,冷冻仅影响买入操作的时机[^1][^2] ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值