「动态规划」买卖股票的最佳时机,如何处理多笔交易?

2024-06-15 05:12

本文主要是介绍「动态规划」买卖股票的最佳时机,如何处理多笔交易?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

188. 买卖股票的最佳时机 IVicon-default.png?t=N7T8https://leetcode.cn/problems/best-time-to-buy-and-sell-stock-iv/description/

给你一个整数数组prices和一个整数k,其中prices[i]是某支给定的股票在第i天的价格。设计一个算法来计算你所能获取的最大利润。你最多可以完成k笔交易。也就是说,你最多可以买k次,卖k次。注意:你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。

  1. 输入:k = 2,prices = [2,4,1],输出:2,解释:在第1天(股票价格 = 2)的时候买入,在第2天(股票价格 = 4)的时候卖出,这笔交易所能获得利润 = 4 - 2 = 2。
  2. 输入:k = 2,prices = [3,2,6,5,0,3],输出:7,解释:在第2天(股票价格 = 2)的时候买入,在第3天(股票价格 = 6)的时候卖出,这笔交易所能获得利润 = 6 - 2 = 4。随后,在第5天(股票价格 = 0)的时候买入,在第6天(股票价格 = 3)的时候卖出,这笔交易所能获得利润 = 3 - 0 = 3。

提示:1 <= k <= 100,1 <= prices.length <= 1000,0 <= prices[i] <= 1000。


我们用动态规划的思想来解决这个问题。

确定状态表示:根据经验和题目要求,我们把状态细分为:

  • 我们用f[i][j]表示:在第i天结束时,处于买入状态下,总共交易j次,此时的最大利润。
  • 我们用g[i][j]表示:在第i天结束时,处于卖出状态下,总共交易j次,此时的最大利润。

解释一下上面出现的名词。如果我们手里有股票,我们称当前处于买入状态下;如果我们手里没有股票,我们称当前处于卖出状态下。一次完整的买入持有到卖出称为一笔交易,也就是说,一开始的交易次数为0,在每次卖出时交易次数加1。每次买入股票会让利润减少股票在当天的价格,卖出股票会让利润增加股票在当天的价格。在状态表示中,f和g分别表示买入和卖出状态,i表示天数,j表示交易次数,f[i][j]和g[i][j]表示最大利润。

推导状态转移方程:我们需要考虑最近的一步,即第i - 1天的状态和交易次数。

首先考虑f[i][j],即在第i天结束时处于买入状态下,且交易了j次。

  • 如果在第i - 1天结束时,处于买入状态下,且交易了j次,此时的利润是f[i - 1][j],那么只需要在第i天什么都不做,在第i天结束时,依然处于买入状态下,且交易了j次,利润不变,依然是f[i - 1][j]。
  • 如果在第i - 1天结束时,处于卖出状态下,且交易了j次,此时的利润是g[i - 1][j],那么只需要在第i天买入股票,在第i天结束时,就会处于买入状态下,且交易了j次,利润减少股票在第i天的价格,即g[i - 1][j] - prices[i]。

由于f[i][j]表示最大利润,所以取上面2种情况的较大值,即f[i][j] = max(f[i - 1][j], g[i - 1][j] - prices[i])。

接着考虑g[i][j],即在第i天结束时处于卖出状态下,且交易了j次。

  • 如果在第i - 1天结束时,处于买入状态下,且交易了j - 1次,此时的利润是f[i - 1][j - 1],那么只需要在第i天卖出股票,在第i天结束时,就会处于卖出状态下,交易次数加1,即交易了j次,利润增加股票在第i天的价格,即f[i - 1][j - 1] + prices[i]。
  • 如果在第i - 1天结束时,处于卖出状态下,且交易了j次,此时的利润是g[i - 1][j],那么只需要在第i天什么都不做,在第i天结束时,依然处于卖出状态下,且交易了j次,利润不变,依然是g[i - 1][j]。

由于g[i][j]表示最大利润,所以取上面2种情况的较大值,即g[i][j] = max(f[i - 1][j - 1] + prices[i], g[i - 1][j])。

综上所述:f[i][j] = max(f[i - 1][j], g[i - 1][j] - prices[i]),g[i][j] = max(f[i - 1][j - 1] + prices[i], g[i - 1][j])

初始化:根据状态转移方程,

  • 计算f[i][j]时,当i = 0时会越界。
  • 计算g[i][j]时,当i = 0或j = 0时会越界。

所以,我们要初始化相应的位置。容易想到:

  • f[0][0]表示在第0天结束时,处于买入状态下,此时的最大利润。一开始利润是0,在第0天买入股票,显然f[0][0] = -prices[0]。
  • g[0][0]表示在第0天结束时,处于卖出状态下,此时的最大利润。一开始利润是0,在第0天什么都不做,显然g[0][0] = 0。

接着考虑f[0][j],其中j > 0。j > 0说明交易次数至少是1次,也就是说在第0天一定做出了买入并且立刻卖出股票的操作。然而这种操作是没有意义的,因为浪费了交易次数,并不会增加最大利润。观察状态转移方程,发现不管是f[i][j]还是g[i][j],最终都是对2个值求max。要想不影响到计算结果,我们要对f[0][j],其中j > 0的位置的值都初始化为-∞。同理g[0][j],其中j > 0的位置的值也要初始化为-∞。考虑到状态转移方程中,有g[i - 1][j] - prices[i]这样有溢出风险的计算,所以不能简单地用INT_MIN表示-∞,而要用-0x3f3f3f3f。

再考虑g[i][0],其中i > 0。观察状态转移方程:g[i][j] = max(f[i - 1][j - 1] + prices[i], g[i - 1][j])。为什么g[i][0],其中i > 0的位置会越界呢?因为方程中含有f[i - 1][j - 1],当j = 0时,j - 1 = -1,不存在交易次数为-1的情况。所以,我们需要判断一下,当j - 1 = -1时,这种情况不存在,相当于求max的2项中,前一项不存在,那么就只剩下后一项,即g[i - 1][j],即此时g[i][j] = g[i - 1][j]。只有当j - 1 >= 0时,求max的2项都有意义,此时才计算g[i][j] = max(f[i - 1][j - 1] + prices[i], g[i - 1][j])。也就是说,先让g[i][j] = g[i - 1][j],接着判断j - 1是否大于-1,即j是否大于0,如果判断成立,再让g[i][j] = max(g[i][j], f[i - 1][j - 1] + prices[i])。

综上所述:初始化需要注意以下几点:f[0][0] = -prices[0];g[0][0] = 0;f[0][j] = g[0][j] = -0x3f3f3f3f,其中j > 0;当i > 0时,先让g[i][j] = g[i - 1][j],接着判断j - 1是否大于-1,即j是否大于0,如果判断成立,再让g[i][j] = max(g[i][j], f[i - 1][j - 1] + prices[i])。只需做到以上几点,就不会越界。

填表顺序:观察状态转移方程,显然我们要沿着i和j增大的方向同时填f表和g表

返回值:假设总共有n天,最多完成k笔交易。对于第i天,i的范围是[0, n - 1]。根据题目要求,我们要返回的是最后一天结束后的最大利润,即第n - 1天结束后的最大利润。可以确定,如果要求最大利润,第n - 1天结束后一定要处于卖出状态下,否则在第n - 1天卖出股票可以获得更多利润。另外,并不确定第n - 1天结束后的交易次数。根据状态表示,我们要返回的是g[n - 1][j]的最大值,其中j的范围是[0, k]

细节问题:由于i的范围是[0, n - 1],j的范围是[0, k],所以f表和g表的规模都是n x (k + 1)。另外,交易次数不会超过天数的一半,所以要先计算k = min(k, n / 2)

时间复杂度:O(N^2),空间复杂度:O(N^2)。最坏情况是k刚好是n的一半。

class Solution {
public:int maxProfit(int k, vector<int>& prices) {const int INF = 0x3f3f3f3f;int n = prices.size();// 交易次数不会超过天数的一半k = min(k, n / 2);// 创建dp表vector<vector<int>> f(n, vector<int>(k + 1, -INF));auto g = f;// 初始化f[0][0] = -prices[0];g[0][0] = 0;// 填表for (int i = 1; i < n; i++) {for (int j = 0; j <= k; j++) {f[i][j] = max(f[i - 1][j], g[i - 1][j] - prices[i]);g[i][j] = g[i - 1][j];if (j > 0) {g[i][j] = max(g[i][j], f[i - 1][j - 1] + prices[i]);}}}// 返回结果return *max_element(g[n - 1].begin(), g[n - 1].end());}
};

123. 买卖股票的最佳时机 IIIicon-default.png?t=N7T8https://leetcode.cn/problems/best-time-to-buy-and-sell-stock-iii/

给定一个数组,它的第i个元素是一支给定的股票在第i天的价格。设计一个算法来计算你所能获取的最大利润。你最多可以完成两笔交易。注意:你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。

  1. 输入:prices = [3,3,5,0,0,3,1,4],输出:6,解释:在第4天(股票价格 = 0)的时候买入,在第6天(股票价格 = 3)的时候卖出,这笔交易所能获得利润 = 3 - 0 = 3 。随后,在第7天(股票价格 = 1)的时候买入,在第8天 (股票价格 = 4)的时候卖出,这笔交易所能获得利润 = 4 - 1 = 3。
  2. 输入:prices = [1,2,3,4,5],输出:4,解释:在第1天(股票价格 = 1)的时候买入,在第5天 (股票价格 = 5)的时候卖出,这笔交易所能获得利润 = 5 - 1 = 4。注意你不能在第1天和第2天接连购买股票,之后再将它们卖出。因为这样属于同时参与了多笔交易,你必须在再次购买前出售掉之前的股票。
  3. 输入:prices = [7,6,4,3,1],输出:0,解释:在这个情况下,没有交易完成,所以最大利润为0。
  4. 输入:prices = [1],输出:0

提示:1 <= prices.length <= 10^5,0 <= prices[i] <= 10^5。


这道题是上道题在k = 2时的特殊情况,我们只需要复用上道题的代码就行了。当然,感兴趣的话,你也可以用动态规划的思想来分析分析。

class Solution {
public:int maxProfit(vector<int>& prices) { return maxProfit(2, prices); }private:int maxProfit(int k, vector<int>& prices) {const int INF = 0x3f3f3f3f;int n = prices.size();// 交易次数不会超过天数的一半k = min(k, n / 2);// 创建dp表vector<vector<int>> f(n, vector<int>(k + 1, -INF));auto g = f;// 初始化f[0][0] = -prices[0];g[0][0] = 0;// 填表for (int i = 1; i < n; i++) {for (int j = 0; j <= k; j++) {f[i][j] = max(f[i - 1][j], g[i - 1][j] - prices[i]);g[i][j] = g[i - 1][j];if (j > 0) {g[i][j] = max(g[i][j], f[i - 1][j - 1] + prices[i]);}}}// 返回结果return *max_element(g[n - 1].begin(), g[n - 1].end());}
};

这篇关于「动态规划」买卖股票的最佳时机,如何处理多笔交易?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

电脑提示xlstat4.dll丢失怎么修复? xlstat4.dll文件丢失处理办法

《电脑提示xlstat4.dll丢失怎么修复?xlstat4.dll文件丢失处理办法》长时间使用电脑,大家多少都会遇到类似dll文件丢失的情况,不过,解决这一问题其实并不复杂,下面我们就来看看xls... 在Windows操作系统中,xlstat4.dll是一个重要的动态链接库文件,通常用于支持各种应用程序

SQL Server数据库死锁处理超详细攻略

《SQLServer数据库死锁处理超详细攻略》SQLServer作为主流数据库管理系统,在高并发场景下可能面临死锁问题,影响系统性能和稳定性,这篇文章主要给大家介绍了关于SQLServer数据库死... 目录一、引言二、查询 Sqlserver 中造成死锁的 SPID三、用内置函数查询执行信息1. sp_w

Java对异常的认识与异常的处理小结

《Java对异常的认识与异常的处理小结》Java程序在运行时可能出现的错误或非正常情况称为异常,下面给大家介绍Java对异常的认识与异常的处理,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参... 目录一、认识异常与异常类型。二、异常的处理三、总结 一、认识异常与异常类型。(1)简单定义-什么是

Golang 日志处理和正则处理的操作方法

《Golang日志处理和正则处理的操作方法》:本文主要介绍Golang日志处理和正则处理的操作方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录1、logx日志处理1.1、logx简介1.2、日志初始化与配置1.3、常用方法1.4、配合defer

springboot加载不到nacos配置中心的配置问题处理

《springboot加载不到nacos配置中心的配置问题处理》:本文主要介绍springboot加载不到nacos配置中心的配置问题处理,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录springboot加载不到nacos配置中心的配置两种可能Spring Boot 版本Nacos

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可

Mybatis嵌套子查询动态SQL编写实践

《Mybatis嵌套子查询动态SQL编写实践》:本文主要介绍Mybatis嵌套子查询动态SQL编写方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、实体类1、主类2、子类二、Mapper三、XML四、详解总结前言MyBATis的xml文件编写动态SQL

python web 开发之Flask中间件与请求处理钩子的最佳实践

《pythonweb开发之Flask中间件与请求处理钩子的最佳实践》Flask作为轻量级Web框架,提供了灵活的请求处理机制,中间件和请求钩子允许开发者在请求处理的不同阶段插入自定义逻辑,实现诸如... 目录Flask中间件与请求处理钩子完全指南1. 引言2. 请求处理生命周期概述3. 请求钩子详解3.1

Python处理大量Excel文件的十个技巧分享

《Python处理大量Excel文件的十个技巧分享》每天被大量Excel文件折磨的你看过来!这是一份Python程序员整理的实用技巧,不说废话,直接上干货,文章通过代码示例讲解的非常详细,需要的朋友可... 目录一、批量读取多个Excel文件二、选择性读取工作表和列三、自动调整格式和样式四、智能数据清洗五、