代码随想录刷题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中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

Java实现自定义table宽高的示例代码

《Java实现自定义table宽高的示例代码》在桌面应用、管理系统乃至报表工具中,表格(JTable)作为最常用的数据展示组件,不仅承载对数据的增删改查,还需要配合布局与视觉需求,而JavaSwing... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

Go语言代码格式化的技巧分享

《Go语言代码格式化的技巧分享》在Go语言的开发过程中,代码格式化是一个看似细微却至关重要的环节,良好的代码格式化不仅能提升代码的可读性,还能促进团队协作,减少因代码风格差异引发的问题,Go在代码格式... 目录一、Go 语言代码格式化的重要性二、Go 语言代码格式化工具:gofmt 与 go fmt(一)

HTML5实现的移动端购物车自动结算功能示例代码

《HTML5实现的移动端购物车自动结算功能示例代码》本文介绍HTML5实现移动端购物车自动结算,通过WebStorage、事件监听、DOM操作等技术,确保实时更新与数据同步,优化性能及无障碍性,提升用... 目录1. 移动端购物车自动结算概述2. 数据存储与状态保存机制2.1 浏览器端的数据存储方式2.1.

基于 HTML5 Canvas 实现图片旋转与下载功能(完整代码展示)

《基于HTML5Canvas实现图片旋转与下载功能(完整代码展示)》本文将深入剖析一段基于HTML5Canvas的代码,该代码实现了图片的旋转(90度和180度)以及旋转后图片的下载... 目录一、引言二、html 结构分析三、css 样式分析四、JavaScript 功能实现一、引言在 Web 开发中,

Python如何去除图片干扰代码示例

《Python如何去除图片干扰代码示例》图片降噪是一个广泛应用于图像处理的技术,可以提高图像质量和相关应用的效果,:本文主要介绍Python如何去除图片干扰的相关资料,文中通过代码介绍的非常详细,... 目录一、噪声去除1. 高斯噪声(像素值正态分布扰动)2. 椒盐噪声(随机黑白像素点)3. 复杂噪声(如伪

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部