news 2026/5/4 2:26:40

代码随想录算法训练营第三十五天 | 121. 买卖股票的最佳时机、122.买卖股票的最佳时机II、123.买卖股票的最佳时机III

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
代码随想录算法训练营第三十五天 | 121. 买卖股票的最佳时机、122.买卖股票的最佳时机II、123.买卖股票的最佳时机III

代码随想录算法训练营第三十五天任务

  • 121. 买卖股票的最佳时机
  • 122.买卖股票的最佳时机II
  • 123.买卖股票的最佳时机III

121. 买卖股票的最佳时机

题目链接:121. 买卖股票的最佳时机
贪心思路:前期尽可能地低价买入,后期尽可能地高价卖出。

classSolution{public:intmaxProfit(vector<int>&prices){intlow=INT_MAX;intprofit=0;for(inti=0;i<prices.size();++i){low=min(low,prices[i]);// 寻找低点profit=max(profit,prices[i]-low);}returnprofit;}};

时间复杂度:O(n)
空间复杂度:O(1)

动态规划:

  1. 确定dp数组的下标及其含义
    dp[i][0]:表示第i天持有股票拥有的最多金额。(持有不代表当天买入,还有可能前几天买入,没买出,一直持有)
    dp[i][1]:表示第i天不持有股票拥有的最多金额。

  2. 确定递推公式
    第i天持有股票,由 “第i天之前持有股票” 和 “第i天买入股票”两种状态推导而来: dp[i][0] = max(dp[i-1][0], - prices[i])
    另外注意:题目要求是一次交易,所以第i天买入股票不能由dp[i-1][1]-prices[i] 而来。
    第i天不持有股票,由 “第i天之前不持有股票” 和 “第i天卖出股票”两种状态推导而来: dp[i][1] = max(dp[i-1][1], dp[i-1][0] + prices[i])

  3. 初始化
    由递推公式可知:第i天由第i-1天推导而来
    dp[0][1]: 表示第0天不持有股票, 所以dp[0][1] = 0
    dp[0][0]: 表示第0天持有股票,所以dp[0][0] = -prices[0]

  4. 确定遍历顺序
    从前往后

  5. 举例推导
    输入:[7,1,5,3,6,4]

idp[i][0]dp[i][1]
0-70
1-10
2-14
3-14
4-15
5-15
classSolution{public:intmaxProfit(vector<int>&prices){vector<vector<int>>dp(prices.size(),vector<int>(2,0));dp[0][0]-=prices[0];// 表示第0天持有股票dp[0][1]=0;// 表示第0天不持有股票for(inti=1;i<prices.size();++i){dp[i][0]=max(dp[i-1][0],-prices[i]);dp[i][1]=max(dp[i-1][1],dp[i-1][0]+prices[i]);}returndp[prices.size()-1][1];}};

时间复杂度:O(n)
空间复杂度:O(n)
还可以用翻滚数组,使空间复杂度为O(1)

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

题目链接:122.买卖股票的最佳时机II
这道题和上一道题的区别就在于可以多次交易。
由上述动规五步曲可知,只一点不同,就是 第i天持有股票,由 “第i天之前持有股票” 和 “第i天买入股票”两种状态推导而来: dp[i][0] = max(dp[i-1][0], dp[i-1][1] - prices[i]) 第i天买入股票可由前一天不持有股票金额推导而来。

classSolution{public:intmaxProfit(vector<int>&prices){vector<vector<int>>dp(prices.size(),vector<int>(2,0));dp[0][0]-=prices[0];// 表示第0天持有股票dp[0][1]=0;// 表示第0天不持有股票for(inti=1;i<prices.size();++i){dp[i][0]=max(dp[i-1][0],dp[i-1][1]-prices[i]);// 与 121. 买卖股票的最佳时机不同之处dp[i][1]=max(dp[i-1][1],dp[i-1][0]+prices[i]);}returndp[prices.size()-1][1];}};

时间复杂度:O(n)
空间复杂度:O(n)
还可以用翻滚数组,使空间复杂度为O(1)

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

题目链接:123.买卖股票的最佳时机III
这道题要求 最多可以完成 两笔 交易。这个限制就像背包一样,不超过。
想不出来,看题解了,原来是分状态。
动规5步曲安排!

  1. 确定dp数组的下标及其含义
    每天有5种状态:

    状态含义
    0不操作
    1第一次持有
    2第一次不持有
    3第二次持有
    4第二次不持有

    dp[i][j]表示第 i 天的第 j 种状态下的最大金额。

  2. 确定递推公式
    dp[i][1] : 第 i 天第一次持有股票,由 “第 i 天之前第一次持有股票” 和 “第 i 天买入股票”两种状态推导而来。
    dp[i][1] = max(dp[i-1][1], dp[i-1][0] - prices[i])
    dp[i][2]: 第 i 天第一次不持有股票,由 “第 i 天之前第一次不持有股票” 和 “第 i 天卖出股票”两种状态推导而来。
    dp[i][2] = max(dp[i-1][2], dp[i-1][1] + prices[i])
    dp[i][3]: 第 i 天第二次持有股票,由 “第 i 天之前第二次持有股票” 和 “第 i 天买入股票”两种状态推导而来。
    dp[i][3] = max(dp[i-1][3], dp[i-1][2] - prices[i])
    dp[i][4]: 第 i 天第二次不持有股票,由 “第 i 天之前第二次不持有股票” 和 “第 i 天卖出股票”两种状态推导而来。
    dp[i][4] = max(dp[i-1][4], dp[i-1][3] + prices[i])

  3. 初始化
    dp[0][0]: 表示第0天不操作,dp[0][0] = 0.
    dp[0][1]: 表示第0天第一次持有, dp[0][1] = -prices[0]
    dp[0][2]: 表示第0天第一次不持有, dp[0][2] = 0
    dp[0][3]: 表示第0天第二次持有, dp[0][3] = -prices[0]
    dp[0][4]: 表示第0天第二次不持有, dp[0][4] = 0
    dp[i][0] : 表示第 i 天什么都不操作,不是第一次持有/不持有,第二次持有/不持有的任何一个状态,无操作,dp[i][0] = 0. 这列数后续也没用到。

  4. 确定遍历顺序
    从前往后

  5. 举例推导
    prices = [3,3,5,0,0,3,1,4]

    iprices[i]dp[i][0]dp[i][1]dp[i][2]dp[i][3]dp[i][4]
    030-30-30
    130-32-32
    250-32-32
    3000222
    4000222
    5300325
    6100325
    7400426
classSolution{public:intmaxProfit(vector<int>&prices){vector<vector<int>>dp(prices.size(),vector<int>(5,0));// dp[0][0] = 0; // 表示第0天不操作dp[0][1]=-prices[0];// 表示第0天第一次持有// dp[0][2] = 0; // 表示第0天第一次不持有dp[0][3]=-prices[0];// 表示第0天第二次持有// dp[0][4] = 0; // 表示第0天第二次不持有for(inti=1;i<prices.size();++i){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]);}returndp[prices.size()-1][4];}};

时间复杂度:O(n)
空间复杂度:O(n)
还可以用翻滚数组,使空间复杂度为O(1)

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/2 10:43:14

浅谈JS在挖掘CNVD通用漏洞中的渗透思路,附实战案例教程!

前言&#xff1a;本文中涉及到的相关技术或工具仅限技术研究与讨论&#xff0c;严禁用于非法用途&#xff0c;否则产生的一切后果自行承担&#xff0c;如有侵权请联系。 本文纯干货详细记述了利用js在挖掘cnvd通用漏洞中的思路以及附带的实战案例。 一、前言 JS渗透测试是一种针…

作者头像 李华
网站建设 2026/4/30 14:57:38

麒麟系统开机自动打开浏览器并全屏

首先要自动登录 在设置—>账户—>账户信息&#xff0c;选择开机自动登录配置浏览器打开时打开指定页面 浏览器的设置中&#xff0c;选择启动时打开主页&#xff0c;主页地址设置为你想要打开的网页地址设置浏览器开机自启并全屏 开机自启 首先找到浏览器的.desktop文件&a…

作者头像 李华
网站建设 2026/4/24 8:40:38

AI 知识科普|部署 AI 模型,涉及哪些机房层面的改造?

部署 AI 模型&#xff0c;涉及哪些机房层面的改造&#xff1f; 在构建 AI 基础设施时&#xff0c;除了基础设施本身的建设&#xff0c;不少企业也需要改造现有数据中心机房&#xff0c;以满足 AI 运行在电力、冷却和机房空间等方面的特殊需求。 供电系统升级 AI 模型往往采用…

作者头像 李华
网站建设 2026/5/1 5:53:12

Flutter 图片内存优化指南(完整版)

欢迎大家加入开源鸿蒙跨平台开发者社区&#xff0c;一起共建开源鸿蒙跨平台生态。 Flutter 图片内存优化指南&#xff08;完整版&#xff09; 在 Flutter 应用中&#xff0c;图片资源往往是内存消耗的主要来源。特别是在处理高分辨率图片或大量图片时&#xff0c;不合理的内存…

作者头像 李华