代码随想录算法训练营打卡Day51 | LeetCode309 最佳买卖股票的时机含冷冻期、LeetCode714 买卖股票的最佳时机含手续费

摘要

  • 有些动态规划的题目的难点在于如何划分状态和这些状态之间如何进行转移,列出可能的状态以及转移到这些状态的可能,是定义dp数组及数组下标、推导递推公式的关键。

  • 画出简单的状态转移示意图,有助于清楚地划分状态以及模拟状态转移的过程,对如何定义dp数组以及推导状态转移方程有一定启发。

  • 对于初始化中的非法状态,不要死抠dp数组的定义,只要初始值能让递推公式能正确计算即可。

  • 有一部分可以使用动态规划解决的问题,如果问题具有贪心选择性质,也可以用贪心法解决,贪心法是动态规划的特例,贪心法的效率往往比动态规划要高。而动态规划往往是一类问题的通解,代码的复用性和可扩展性更好。

LeetCode309 最佳买卖股票的时机含冷冻期

309. 最佳买卖股票时机含冷冻期 - 力扣(Leetcode)

  • 这道题目添加了“冷冻期”这个规则,需要更精细地划分每一天的状态。

    • 首先,本题不限制买卖股票地次数,所以不需要记录每天买卖股票的次数。但要先买入股票才能卖出股票,至少需要记录每天是否持有股票。

    • 但由于“冷冻期”的限制,不能再卖出股票后的第二天买入股票,所以卖出股票的第二天一定不能持有股票,这和其他时候没有持有股票的是不一样的。非“冷静期”时可以选择是否买入股票,而“冷静期”时不能选择买入股票。都是不持有股票,但应该区分冷静期和非冷静期。

    • 对一天的状态尝试进行划分,一天可能有如下4种状态

      1. 今天持有股票
      2. 今天卖出股票
      3. 今天是冷冻期,不能买入股票
      4. 今天不是冷冻期,且未持有股票
    • 然后要分析这些状态相互之间如何进行转移

      • 今天持有股票:
        1. 前一天持有股票且不卖出,那今天也是持有股票
        2. 前一天卖出了股票,今天是冷冻期,不能买入股票
        3. 前一天是冷冻期,那今天正好可以买入股票,今天买入股票
        4. 前一天不是冷冻期且未持有股票,那今天买入股票
      • 今天卖出股票
        1. 前一天一定要持有股票,今天才能卖出
      • 今天是冷冻期
        1. 只有前一天卖出了股票,今天才会是冷冻期
      • 今天不是冷冻期,且未持有股票
        1. 前一天是冷冻期,今天正好解冻,但是今天不买入股票
        2. 前一天不是冷冻期,也未持有股票,今天不买入股票
    • 可以得到这样的状态转移图

image.png
  • 确定dp数组及数组下标的含义:dp[i][j]为到第i天时能获得的最大金额,j代表第i天处于什么状态:j==0为“今天持有股票”,j==1为“今天卖出股票”,j==2为“今天是冷静期”,j==3为“今天不是冷静期且未持有股票”。

  • 确定状态转移方程,根据之前对状态转移的分析

    • 今天持有股票,dp[i][0]
      1. 前一天持有股票且不卖出,那今天也是持有股票dp[i][0]=dp[i-1][0]
      2. 前一天卖出了股票,今天是冷冻期,不能买入股票,不能转移到这个状态
      3. 前一天是冷冻期,那今天正好可以买入股票,今天买入股票dp[i-1][0]=dp[i-1][2]-prices[i]
      4. 前一天不是冷冻期且未持有股票,那今天买入股票dp[i-1][0]=dp[i-1][3]-prices[i]
    • 今天卖出股票
      1. 前一天一定要持有股票,今天才能卖出dp[i][1]=dp[i-1][0]+prices[i]
    • 今天是冷静期
      1. 只有前一天卖出了股票,今天才会是冷冻期dp[i][2]=dp[i-1][1]
    • 今天不是冷静期且未持有股票
      1. 前一天是冷冻期,今天正好解冻,但是今天不买入股票dp[i][3]=dp[i-1][2]
      2. 前一天不是冷冻期,也未持有股票,今天不买入股票dp[i][3]=dp[i-1][3]
  • 确定初始状态,初始化dp数组,假设初始金额为0

    • 0天持有股票就是第0天买入,dp[0][0]=0-prices[i]
    • 0天卖出股票,看成第0天买入再卖出(虽然题目不允许这样),也可以看成是为了要正确计算下一天处于冷静期时的dp值,dp[0][1]应该初始化成0dp[0][1]=0
    • 0天是冷静期,看成第0天买入再卖出,第0天也不允许再进行买卖了,也可以看成是为了要正确计算下一天未持有股票的dp值,dp[0][2]应该初始化成0dp[0][2]=0
    • 0天不是冷静期且未持有股票,实际上就是第0天什么也不做,dp[0][3]=0
  • 每天的各个状态的更新都依赖于上一天的状态,所以i应该从小到大遍历。

题解代码如下

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        vector<vector<int>> dp(prices.size(), vector<int>(4, 0));
        dp[0][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][2], dp[i - 1][3]) - prices[i]);
            dp[i][1] = dp[i - 1][0] + prices[i];
            dp[i][2] = dp[i - 1][1];
            dp[i][3] = max(dp[i - 1][2], dp[i - 1][3]);
        }

        return max(dp[prices.size() - 1][1], max(dp[prices.size() - 1][2], dp[prices.size() - 1][3]));
    }
};
  • 最后肯定是手上未持有股票时,持有的金额最大,而未持有股票实际上就对应着下标j[1-3]的状态,从这三种中去最大值即可。

LeetCode714 买卖股票的最佳时机含手续费

714. 买卖股票的最佳时机含手续费 - 力扣(Leetcode)

  • 这道题目和 122. 买卖股票的最佳时机 II - 力扣(Leetcode) 也只是在递推公式上有区别,递推公式中需要加入手续费的计算。在 代码随想录算法训练营打卡Day49 中,已经用动态规划的思路较详细的分析了这道题,所以这次就简单的过一遍。

  • dp数组及数组下标的含义:dp[i][j]表示到第i天时买卖股票能获得的最大利润,j==0表示当天持有股票,j==1当天未持有股票。

  • 状态转移方程,假设初始金额为0

    • 对于第i天,

      • 如果持有股票,说明在第i天或之前的某一天买入了股票。
        • 如果是在第i天买入了股票,至少第i-1天时是未持有股票的,根据dp数组的定义,第i-1天持有的金额是dp[i-1][1],那么第i天时持有的金额为dp[i][0] = dp[i-1][1] - prices[i]
        • 如果在之前的某一天买入了股票,现在还应该继续持有股票,所以 dp[i][0] = dp[i - 1][0],保持着持有之前买入的股票的状态。
      • 如果不持有股票,说明在第i天或之前的某一天卖出了股票。
        • 如果是在第i天买入了股票,至少第i-1天时是持有股票的,根据dp数组的定义,第i-1天持有的金额是dp[i-1][0],然后不要忘记卖出股票时要支付手续费,那么第i天时持有的金额为dp[i][1] = dp[i - 1][0] + prices[i] - fee
        • 如果在之前的某一天卖出了股票,第i天还不应该买入股票,所以,保持之前的状态即可 dp[i][1] = dp[i - 1][1]
  • 初始化dp数组,第0天持有股票就是第0天买入股票,dp[0][0]=0-prices[i];第0天未持有股票就是什么也不做,dp[0][0]=0dp数组其他的值初始化成不影响递推公式计算结果的值即可,既然假设初始金额为0,那初始化成0就可以。

  • 遍历顺序,就是今天的状态依赖上一天的状态,i从小到大遍历

class Solution {
public:
    int maxProfit(vector<int>& prices, int fee) {
        vector<vector<int>> dp(prices.size(), {0, 0});
        dp[0][0] = 0 - prices[0];
        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]);
            dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] + prices[i] - fee);
        }

        return dp[prices.size() - 1][1];
    }
};
  • 似乎股票问题相当一部分能用贪心法解决,先写完这一轮的动态规划再回头看贪心法。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容