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

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.mzph.cn/bicheng/27429.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于钽电容器的作用、优缺点、选型指南及故障诊断方法等介绍

钽电容器&#xff0c;全称为钽电解电容器&#xff0c;是一种以金属钽作为介质材料的电解电容器。与传统的电解电容器不同&#xff0c;钽电容器不使用液体电解质&#xff0c;而是利用钽氧化物&#xff08;五氧化二钽&#xff09;作为固态电解质&#xff0c;这使得它们具有更高的…

取证工作: SysTools SQL Log Analyzer, 完整的 SQL Server 日志取证分析

天津鸿萌科贸发展有限公司是 Systools 系列软件的授权代理商。 SysTools SQL Log Analyzer 是 Systools 取证工具系列之一&#xff0c;用于调查 SQL Server 事务日志&#xff0c;以对数据库篡改进行取证分析。 什么是 SQL Server 事务日志&#xff1f; 在深入研究 SQL 事务日…

失眠焦虑?这些小妙招助你重拾宁静之夜

在这个快节奏的时代&#xff0c;失眠与焦虑似乎成了不少人的“常客”。每当夜幕降临&#xff0c;躺在床上却辗转反侧&#xff0c;思绪万千&#xff0c;仿佛整个世界的喧嚣都涌入了脑海。&#x1f4ad; 其实&#xff0c;放松心情&#xff0c;调整心态&#xff0c;是缓解失眠焦虑…

Java | Leetcode Java题解之第151题反转字符串中的单词

题目&#xff1a; 题解&#xff1a; class Solution {public String reverseWords(String s) {StringBuilder sb trimSpaces(s);// 翻转字符串reverse(sb, 0, sb.length() - 1);// 翻转每个单词reverseEachWord(sb);return sb.toString();}public StringBuilder trimSpaces(S…

flask基础知识1

目录 1.介绍 2.体验一下 3.配置参数&#xff1a; 4.路由和URL 1.路由 2.动态路由&#xff1a; 自定义转换器&#xff1a; 3.使用自定义转换器 5.url_for函数 6.request参数 7.处理响应&#xff1a; 1.重定向&#xff1a; 2.返回json数据&#xff1a; 3.返回模板&…

el-table 多选回显,分页回显

实现el-table多选分页回显功能&#xff0c;左侧是分页的数据源&#xff0c;右侧是选择后的人员数据&#xff0c;切换下一页&#xff0c;选中的数据会在左侧表格回显。 实现&#xff1a; <template><el-dialog :title"title" :visible.sync"show"…

java:spring使用【@ImportResource】导入一个xml里面定义的bean

# 项目代码资源&#xff1a; 可能还在审核中&#xff0c;请等待。。。 https://download.csdn.net/download/chenhz2284/89434148 # 项目代码 【pom.xml】 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-start…

长亭培训加复习安全产品类别

下面这个很重要参加hw时要问你用的安全产品就有这个 检测类型产品 偏审计 安全防御类型 EDR类似于杀毒软件 安全评估 任何东西都要经过这个机械勘察才能上线 安全管理平台 比较杂 比较集成 审计 漏扫 评估 合在这一个平台 也有可能只是管理 主机理解为一个电脑 安了终端插件…

Vue39-组件的嵌套

一、嵌套的定义 二、嵌套示例 school组件中&#xff0c;嵌套student组件 局部注册&#xff0c;注册给谁&#xff0c;就在谁的结构里面写&#xff01; vue拿不到<student>组件标签&#xff01;&#xff01;&#xff01; 三、一人之下万人之上的app组件 或者容器里面不用…

6.14作业

使用手动连接&#xff0c;将登录框中的取消按钮使用第二中连接方式&#xff0c;右击转到槽&#xff0c;在该槽函数中&#xff0c;调用关闭函数 将登录按钮使用qt4版本的连接到自定义的槽函数中&#xff0c;在槽函数中判断ui界面上输入的账号是否为"admin"&#xff0…

探索Napier:Kotlin Multiplatform的日志记录库

探索Napier&#xff1a;Kotlin Multiplatform的日志记录库 在现代软件开发中&#xff0c;日志记录是不可或缺的部分&#xff0c;它帮助开发者追踪应用的行为和调试问题。对于Kotlin Multiplatform项目而言&#xff0c;能够在多个平台上统一日志记录的方法显得尤为重要。Napier…

WordPress实时搜索插件Ajax Search Lite,轻松替代默认搜索功能

WordPress自带的默认搜索功能是跳转到搜索结果页&#xff0c;如果你想要实时搜索功能&#xff0c;特别是在问答中心显示搜索功能&#xff0c;那么建议使用这个WordPress实时搜索插件Ajax Search Lite&#xff0c;它可以在文章、页面、自定义类型文章中搜索标题、内容、摘要、自…

XP系统安装Node.js v8.6.0并搭建Vue2开发环境(项目兼容到Vista的IE9浏览器)

下载并安装Node.js v8.6.0 通常我们开发Vue2项目&#xff0c;是通过vue create命令建立Vue2工程&#xff0c;用npm run serve命令启动Vue2网站的。 vue命令是用JavaScript写的&#xff0c;不是用C语言写的&#xff0c;必须要Node.js环境才能运行&#xff0c;由Node.js自带的np…

【Redis】String的常用命令及图解String使用场景

本文将详细介绍 Redis String 类型的常见命令及其使用场景&#xff0c;包括缓存、计数器、共享会话、手机验证码、分布式锁等场景&#xff0c;并且配图和伪代码进一步方便理解和使用。 命令执行效果时间复杂度set key value [key value…]设置key的值是valueO(k),k是键个数get…

『大模型笔记』主成分分析(PCA)解释:简化机器学习中的复杂数据!

主成分分析(PCA)解释:简化机器学习中的复杂数据 文章目录 一. 主成分分析(PCA)解释:简化机器学习中的复杂数据!二. 参考文献一. 主成分分析(PCA)解释:简化机器学习中的复杂数据! 主成分分析(Principal Component Analysis,简称PCA)通过 将大型数据集中的维度减少…

C#开发-集合使用和技巧(二)Lambda 表达式介绍和应用

C#开发-集合使用和技巧 Lambda 表达式介绍和应用 C#开发-集合使用和技巧介绍简单的示例&#xff1a;集合查询示例&#xff1a; 1. 基本语法从主体语句上区分&#xff1a;1. 主体为单一表达式2. 主体是代码块&#xff08;多个表达式语句&#xff09; 从参数上区分1. 带输入参数的…

【数据结构初阶】 --- 单链表

关于链表你应该先了解这些 下图描述了物理模型和逻辑模型&#xff0c;大多数常见的其实是逻辑模型&#xff0c;但这对初学者或者掌握不扎实的同学不太友好&#xff0c;所以这里我重点讲解物理模型&#xff0c;当了解了这些细节&#xff0c;以后做题或是什么就直接画逻辑模型就…

Java优雅统计耗时【工具类】

任务耗时如何优雅的打印&#xff0c;看完本文你就明白了&#xff01;~ import cn.hutool.core.date.StopWatch; import cn.hutool.core.lang.Console;/*** 优雅打印出任务耗时*/ public class Main {public static void main(String[] args) throws Exception{StopWatch stopW…

macOS Sequoia 开发者测试版下载和安装教程

macOS Sequoia 于 2024年6月10日在WWDC 2024 上发布&#xff0c;里面添加了AI、窗口排列、操控iPhone等功能&#xff0c;目前发布的为测试版本&#xff0c;可能很多人不知道怎么去下载安装&#xff0c;现在小编教一下大家怎么安装最新的 macOS Sequoia 开发者测试版。 下载 mac…

2024 年最新使用 Node 搭建QQ开放平台官方 QQ 频道机器人详细教程(更新中)

注册 QQ 开放平台账号 QQ 开放平台是腾讯应用综合开放类平台&#xff0c;包含 QQ 机器人、QQ 小程序、QQ 小游戏 等集成化管理&#xff0c;也就是说你注册了QQ 开放平台&#xff0c;你开发 QQ 机器人还是 QQ 小程序都是在这个平台进行部署上线和管理。 如何注册 QQ 开放平台账…