news 2026/4/23 6:08:32

代码随想录算法训练营第三十五天 | 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/4/23 11:26:34

Dify低代码平台部署大模型时的GPU资源需求分析

Dify低代码平台部署大模型时的GPU资源需求分析 在AI应用开发日益普及的今天&#xff0c;越来越多企业希望通过低代码平台快速构建基于大语言模型&#xff08;LLM&#xff09;的智能服务。Dify正是其中的典型代表——它以可视化界面简化了从模型选择到服务部署的全流程。但当我们…

作者头像 李华
网站建设 2026/4/23 10:09:56

外网访问图形数据库 Neo4j

Neo4j 是一款基于 JAVA 的图数据库&#xff0c;使用原生图存储和检索技术管理来数据。以节点和关系的形式存储&#xff0c;且使用声明式语言 Cypher 语法简洁。有助于处理复杂的互连和查询具有灵活性和扩展性。本文将详细介绍如何在本地安装 Neo4j 以及结合路由侠内网穿透实现外…

作者头像 李华
网站建设 2026/4/23 10:10:17

用LobeChat搭建团队内部知识助手,同时推广GPU算力服务

用LobeChat搭建团队内部知识助手&#xff0c;同时推广GPU算力服务 在一家中型科技公司里&#xff0c;新员工入职三天后仍搞不清差旅报销标准&#xff1b;研发团队的 A100 显卡白天跑训练任务&#xff0c;晚上却安静地“睡觉”&#xff1b;而市场部同事为了查一个产品参数&#…

作者头像 李华
网站建设 2026/4/23 11:26:43

LobeChat会话管理机制揭秘:持久化存储与上下文保持实践

LobeChat会话管理机制揭秘&#xff1a;持久化存储与上下文保持实践 在如今的AI交互场景中&#xff0c;用户早已不再满足于“问一句、答一句”的机械对话。我们期待的是一个能记住上下文、理解角色设定、甚至跨设备延续对话的智能助手——就像和一位真正懂你的同事协作那样自然流…

作者头像 李华
网站建设 2026/4/22 20:08:30

测试循环结构经常踩坑?那些测试老司机们都这样处理~

对于很多小伙伴来说&#xff0c;循环结构是一个既简单又复杂的测试内容。因为&#xff0c;在测试过程中&#xff0c;多次重复循环可能导致内存泄漏&#xff0c;甚至存在边界错误。 因此&#xff0c;在做循环结构测试时&#xff0c;我们一定要重点关注循环过程的正确性。换句话…

作者头像 李华