代码随想录刷题day50|股票买卖的最佳时机III(至多2次买入和卖出)股票买卖的最佳时机IV(至多K次买入和卖出)

本文主要是介绍代码随想录刷题day50|股票买卖的最佳时机III(至多2次买入和卖出)股票买卖的最佳时机IV(至多K次买入和卖出),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • day50学习内容
  • 一、股票买卖的最佳时机III--至多2次买入和卖出
    • 1.1、动态规划五部曲
      • 1.1.1、 确定dp数组(dp table)以及下标的含义
      • 1.1.2、确定递推公式
      • 1.1.3、 dp数组如何初始化
      • 1.1.4、确定遍历顺序
    • 1.2、代码
      • 1.2.1、如何理解这段代码
  • 二、股票买卖的最佳时机IV--至多K次买入和卖出
    • 2.1、动态规划五部曲
      • 2.1.1、 确定dp数组(dp table)以及下标的含义
      • 2.1.2、确定递推公式
      • 2.1.3、 dp数组如何初始化
      • 2.1.4、确定遍历顺序
      • 2.1.5、输出结果
    • 2.2、代码
      • 2.2.1、代码抄的,太难了
  • 总结
    • 1.感想
    • 2.思维导图


day50学习内容

day50主要内容

  • 股票买卖的最佳时机III–至多2次买入和卖出
  • 股票买卖的最佳时机IV–至多K次买入和卖出

声明
本文思路和文字,引用自《代码随想录》


一、股票买卖的最佳时机III–至多2次买入和卖出

123.原题链接

1.1、动态规划五部曲

1.1.1、 确定dp数组(dp table)以及下标的含义

dp[i][j] 代表的是在第 i 天结束时,根据不同的交易状态(即 j),所能达到的最大利润。这个动态规划表通过不断更新这些状态,帮助我们找到在规定的交易次数内(本题为两次)能获得的最大利润。

1.1.2、确定递推公式

对于每一天 i 和每个状态 j,如下计算 dp[i][j]

  1. 没有操作 (j = 0): 这个状态实际上不参与计算,因为我们总是从第一次买入开始计算,但如果要表示它的话,可以认为它是 dp[i][0] = 0

  2. 第一次买入 (j = 1):

dp[i][1]=max(dp[i−1][1],−prices[i])

这意味着第一次买入的最大利润是上一天已经买入的最大利润和今天买入(当前价格的负值,因为我们支付了这么多钱)之间的较大者。

  1. 第一次卖出 (j = 2):
dp[i][2]=max(dp[i−1][2],dp[i−1][1]+prices[i])

第一次卖出的最大利润是上一天卖出的最大利润和今天卖出(即上一天买入的最大利润加上今天的价格)之间的较大者。

  1. 第二次买入 (j = 3):
dp[i][3]=max(dp[i−1][3],dp[i−1][2]−prices[i])

第二次买入的最大利润是上一天第二次买入的最大利润和上一天第一次卖出后再今天买入(上一天卖出的最大利润减去今天的价格)之间的较大者。

  1. 第二次卖出 (j = 4):
dp[i][4]=max(dp[i−1][4],dp[i−1][3]+prices[i])

第二次卖出的最大利润是上一天第二次卖出的最大利润和上一天第二次买入后今天卖出(上一天第二次买入的最大利润加上今天的价格)之间的较大者。

通过这些递推公式,我们能够从第一天遍历到最后一天,同时更新每天的五个状态,最终得到完成两次交易的最大利润,即 dp[n-1][4],其中 n 是给定价格数组的长度。

1.1.3、 dp数组如何初始化

在第一天,我们可以选择买入股票,所以 dp[0][1] 和 dp[0][3] 都被初始化为 -prices[0],
这表示第一天买入股票的成本。其他状态的利润在第一天都是0(因为我们还没有进行任何卖出操作)

1.1.4、确定遍历顺序

从小到大遍历

1.2、代码

class Solution {public int maxProfit(int[] prices) {int len = prices.length;if (prices.length == 0)return 0;/** 定义 5 种状态:* 0: 没有操作, 1: 第一次买入, 2: 第一次卖出, 3: 第二次买入, 4: 第二次卖出*/int[][] dp = new int[len][5];dp[0][1] = -prices[0];dp[0][3] = -prices[0];for (int i = 1; i < len; i++) {dp[i][1] = Math.max(dp[i - 1][1], -prices[i]);dp[i][2] = Math.max(dp[i - 1][2], dp[i - 1][1] + prices[i]);dp[i][3] = Math.max(dp[i - 1][3], dp[i - 1][2] - prices[i]);dp[i][4] = Math.max(dp[i - 1][4], dp[i - 1][3] + prices[i]);}return dp[len - 1][4];}
}

1.2.1、如何理解这段代码

  1. 问题定义:给定一个数组 prices,其中 prices[i] 是第 i 天的股票价格。目标是找出最多进行两次交易(买入和卖出算一次交易)能获得的最大利润。注意,第二次买入必须在第一次卖出之后。

  2. 动态规划状态定义:定义一个二维数组 dp[len][5],其中 len 是给定价格数组的长度。dp[i][j] 表示第 i 天结束时处于五种状态之一的最大利润:

    • j = 0:没有进行任何操作。
    • j = 1:进行了第一次买入操作。
    • j = 2:进行了第一次卖出操作。
    • j = 3:进行了第二次买入操作。
    • j = 4:进行了第二次卖出操作。
  3. 初始化:在第一天,我们可以选择买入股票,所以 dp[0][1]dp[0][3] 都被初始化为 -prices[0],这表示第一天买入股票的成本。其他状态的利润在第一天都是0(因为我们还没有进行任何卖出操作),但代码中省略了这部分初始化,因为Java数组默认初始化为0。

  4. 状态转移方程:从第二天开始,我们根据前一天的状态更新当前天的状态:

    • dp[i][1]:第一次买入的最大利润,可能是前一天已经买入,或者今天才买入,取二者的较大值。
    • dp[i][2]:第一次卖出的最大利润,可能是前一天已经卖出,或者今天卖出(今天卖出的话,利润是前一天买入的利润加上今天的价格)。
    • dp[i][3]:第二次买入的最大利润,可能是前一天已经买入,或者今天买入(今天买入的话,利润是前一天第一次卖出的利润减去今天的价格)。
    • dp[i][4]:第二次卖出的最大利润,可能是前一天已经卖出,或者今天卖出(今天卖出的话,利润是前一天第二次买入的利润加上今天的价格)。
  5. 输出:最终的答案是 dp[len - 1][4],即最后一天结束时,完成两次买卖操作的最大利润。

通过动态规划表来追踪并更新每一天可能的最大利润,从而找到最多进行两次交易所能获得的最大利润。

二、股票买卖的最佳时机IV–至多K次买入和卖出

188.原题链接

2.1、动态规划五部曲

2.1.1、 确定dp数组(dp table)以及下标的含义

  • dp[len][k*2 + 1]:一个二维数组,len 是天数,k*2 + 1 是可能的交易状态数量。由于每次交易包括买入和卖出两个操作,因此共有 k*2 个操作状态,加上一个初始的没有操作状态,总共是 k*2 + 1 状态。
  • 对于 dp[i][j]
    • j 为0时,表示没有进行任何操作。
    • j 为奇数时,表示第 (j+1)/2 次交易持有股票(即买入状态)。
    • j 为偶数时,表示第 j/2 次交易不持有股票(即卖出状态)。

2.1.2、确定递推公式

对于第 i 天(i > 0),遍历所有交易状态 j,根据 j 的奇偶性,分别更新买入状态和卖出状态的最大利润:

  • 买入状态j 为奇数):dp[i][j] = Math.max(dp[i - 1][j], dp[i - 1][j - 1] - prices[i])
    • 如果在第 i 天买入股票,则最大利润是前一天同状态的利润和前一天卖出状态的利润减去今天的价格中的较大值。
  • 卖出状态j 为偶数):dp[i][j] = Math.max(dp[i - 1][j], dp[i - 1][j - 1] + prices[i])
    • 如果在第 i 天卖出股票,则最大利润是前一天同状态的利润和前一天买入状态的利润加上今天的价格中的较大值。

2.1.3、 dp数组如何初始化

对于第一天(i=0):

  • 没有进行任何交易的利润是0,即 dp[0][0] = 0
  • 对于所有的买入状态(j 为奇数),初始化为 -prices[0],因为如果在第一天买入股票,则损失了 prices[0] 的金额。

2.1.4、确定遍历顺序

从小到大遍历

2.1.5、输出结果

最终的结果是 dp[len - 1][k*2],即最后一天,进行完 k 次交易(全部卖出)后的最大利润。

通过这种方法,代码动态地计算在每一天结束时,根据不同交易次数和买卖状态下的最大利润,最终找到在给定交易次数限制下能够获得的最大利润。

2.2、代码

class Solution {public int maxProfit(int k, int[] prices) {if (prices.length == 0) return 0;// [天数][股票状态]// 股票状态: 奇数表示第 k 次交易持有/买入, 偶数表示第 k 次交易不持有/卖出, 0 表示没有操作int len = prices.length;int[][] dp = new int[len][k*2 + 1];for (int i = 1; i < k*2; i += 2) {dp[0][i] = -prices[0];}for (int i = 1; i < len; i++) {for (int j = 0; j < k*2 - 1; j += 2) {dp[i][j + 1] = Math.max(dp[i - 1][j + 1], dp[i - 1][j] - prices[i]);dp[i][j + 2] = Math.max(dp[i - 1][j + 2], dp[i - 1][j + 1] + prices[i]);}}return dp[len - 1][k*2];}
}

2.2.1、代码抄的,太难了

总结

1.感想

  • 2题hard题,都不会写,但是一看题解,好像也不是完全写不来的感觉。。

2.思维导图

本文思路引用自代码随想录,感谢代码随想录作者。

这篇关于代码随想录刷题day50|股票买卖的最佳时机III(至多2次买入和卖出)股票买卖的最佳时机IV(至多K次买入和卖出)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/892927

相关文章

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

nginx-rtmp-module模块实现视频点播的示例代码

《nginx-rtmp-module模块实现视频点播的示例代码》本文主要介绍了nginx-rtmp-module模块实现视频点播,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录预置条件Nginx点播基本配置点播远程文件指定多个播放位置参考预置条件配置点播服务器 192.

CSS自定义浏览器滚动条样式完整代码

《CSS自定义浏览器滚动条样式完整代码》:本文主要介绍了如何使用CSS自定义浏览器滚动条的样式,包括隐藏滚动条的角落、设置滚动条的基本样式、轨道样式和滑块样式,并提供了完整的CSS代码示例,通过这些技巧,你可以为你的网站添加个性化的滚动条样式,从而提升用户体验,详细内容请阅读本文,希望能对你有所帮助...

深入解析Spring TransactionTemplate 高级用法(示例代码)

《深入解析SpringTransactionTemplate高级用法(示例代码)》TransactionTemplate是Spring框架中一个强大的工具,它允许开发者以编程方式控制事务,通过... 目录1. TransactionTemplate 的核心概念2. 核心接口和类3. TransactionT