【LeetCode】动态规划—123. 买卖股票的最佳时机 III(附完整Python/C++代码)

动态规划—123. 买卖股票的最佳时机 III

  • 题目描述
  • 前言
  • 基本思路
    • 1. 问题定义
    • 2. 理解问题和递推关系
      • 状态定义:
      • 状态转移公式:
      • 初始条件:
    • 3. 解决方法
      • 动态规划方法
      • 伪代码:
    • 4. 进一步优化
    • 5. 小总结
  • Python代码
      • Python代码解释
  • C++代码
      • C++代码解释
  • 总结

题目描述

在这里插入图片描述

前言

买卖股票的最佳时机 III 是一个动态规划问题的经典变种。给定一个数组 prices,其中 prices[i] 代表第 i 天的股票价格。你最多可以进行 两次交易(一次交易包含一次买入和一次卖出),目标是通过这两次交易获得最大的利润。该问题的难点在于如何合理规划买卖时机,以获取最大收益。


基本思路

1. 问题定义

给定一个数组 prices,其中 prices[i] 表示股票第 i 天的价格。你最多可以进行两次交易,求这两次交易所能获取的最大利润。注意:两次交易不能交叉进行(必须卖掉一次股票才能再次买入)。

2. 理解问题和递推关系

为了解决该问题,我们可以将其拆解为两个阶段:

  1. 第一次交易:在某天买入并在稍后的一天卖出,获取第一次交易的最大利润。
  2. 第二次交易:在某天(第一次交易之后)再次买入股票并卖出,获取第二次交易的最大利润。

状态定义:

我们可以定义 4 个状态:

  1. first_buy[i]:表示在第 i 天结束时,我们第一次买入股票时的最大收益。
  2. first_sell[i]:表示在第 i 天结束时,我们第一次卖出股票时的最大收益。
  3. second_buy[i]:表示在第 i 天结束时,我们第二次买入股票时的最大收益。
  4. second_sell[i]:表示在第 i 天结束时,我们第二次卖出股票时的最大收益。

状态转移公式:

  1. 第一次买入的状态转移
    f i r s t _ b u y [ i ] = max ⁡ ( f i r s t _ b u y [ i − 1 ] , − p r i c e s [ i ] ) first\_buy[i] = \max(first\_buy[i-1], -prices[i]) first_buy[i]=max(first_buy[i1],prices[i])

    • 要么我们保持之前的状态不变,继续持有第一次买入的股票;
    • 要么在第 i 天以当前价格 prices[i] 买入股票。
  2. 第一次卖出的状态转移
    f i r s t _ s e l l [ i ] = max ⁡ ( f i r s t _ s e l l [ i − 1 ] , f i r s t _ b u y [ i − 1 ] + p r i c e s [ i ] ) first\_sell[i] = \max(first\_sell[i-1], first\_buy[i-1] + prices[i]) first_sell[i]=max(first_sell[i1],first_buy[i1]+prices[i])

    • 要么我们保持之前的状态不变,继续持有卖出后的最大利润;
    • 要么在第 i 天卖出股票,利润为 first_buy[i-1] + prices[i]
  3. 第二次买入的状态转移
    s e c o n d _ b u y [ i ] = max ⁡ ( s e c o n d _ b u y [ i − 1 ] , f i r s t _ s e l l [ i − 1 ] − p r i c e s [ i ] ) second\_buy[i] = \max(second\_buy[i-1], first\_sell[i-1] - prices[i]) second_buy[i]=max(second_buy[i1],first_sell[i1]prices[i])

    • 要么我们保持之前的状态不变,继续持有第二次买入的股票;
    • 要么在第 i 天买入股票,利润为 first_sell[i-1] - prices[i]
  4. 第二次卖出的状态转移
    s e c o n d _ s e l l [ i ] = max ⁡ ( s e c o n d _ s e l l [ i − 1 ] , s e c o n d _ b u y [ i − 1 ] + p r i c e s [ i ] ) second\_sell[i] = \max(second\_sell[i-1], second\_buy[i-1] + prices[i]) second_sell[i]=max(second_sell[i1],second_buy[i1]+prices[i])

    • 要么我们保持之前的状态不变,继续持有第二次卖出的最大利润;
    • 要么在第 i 天卖出股票,利润为 second_buy[i-1] + prices[i]

初始条件:

  • first_buy[0] = -prices[0],第一次买入的初始状态。
  • first_sell[0] = 0,第一次卖出的初始状态。
  • second_buy[0] = -prices[0],第二次买入的初始状态。
  • second_sell[0] = 0,第二次卖出的初始状态。

3. 解决方法

动态规划方法

  1. 定义 4 个状态 first_buy[i]first_sell[i]second_buy[i]second_sell[i] 表示在第 i 天结束时的最大收益。
  2. 通过递推公式更新这些状态。
  3. 最终结果为 second_sell[n-1],即最多进行两次交易后在最后一天的最大利润。

伪代码:

initialize first_buy = -prices[0], first_sell = 0, second_buy = -prices[0], second_sell = 0
for i from 1 to n-1:first_buy = max(first_buy, -prices[i])first_sell = max(first_sell, first_buy + prices[i])second_buy = max(second_buy, first_sell - prices[i])second_sell = max(second_sell, second_buy + prices[i])
return second_sell

4. 进一步优化

  • 空间优化:由于每个状态只依赖于前一天的状态,我们可以将动态规划的 4 个状态用常量来代替,优化空间复杂度为 O(1)

5. 小总结

  • 问题思路:通过定义四个状态,分别对应两次买入和卖出的最大利润,使用动态规划可以有效求解。
  • 时间复杂度:该方法的时间复杂度为 O(n),空间复杂度可以优化为 O(1)

以上就是买卖股票的最佳时机 III问题的基本思路。


Python代码

class Solution:def maxProfit(self, prices: list[int]) -> int:if not prices:return 0# 初始化四个状态first_buy = second_buy = -prices[0]first_sell = second_sell = 0for i in range(1, len(prices)):# 依次更新四个状态first_buy = max(first_buy, -prices[i])first_sell = max(first_sell, first_buy + prices[i])second_buy = max(second_buy, first_sell - prices[i])second_sell = max(second_sell, second_buy + prices[i])# 返回最终状态的最大值return second_sell

Python代码解释

  1. 初始化first_buysecond_buy 初始为 -prices[0]first_sellsecond_sell 初始为 0
  2. 状态转移:通过递推公式依次更新四个状态。
  3. 返回结果:最终返回 second_sell,即最大利润。

C++代码

#include <vector>
#include <algorithm>
using namespace std;class Solution {
public:int maxProfit(vector<int>& prices) {if (prices.empty()) return 0;// 初始化四个状态int first_buy = -prices[0], first_sell = 0;int second_buy = -prices[0], second_sell = 0;for (int i = 1; i < prices.size(); ++i) {// 依次更新四个状态first_buy = max(first_buy, -prices[i]);first_sell = max(first_sell, first_buy + prices[i]);second_buy = max(second_buy, first_sell - prices[i]);second_sell = max(second_sell, second_buy + prices[i]);}// 返回最终状态的最大值return second_sell;}
};

C++代码解释

  1. 初始化first_buysecond_buy 初始为 -prices[0]first_sellsecond_sell 初始为 0
  2. 状态转移:通过递推公式依次更新四个状态。
  3. 返回结果:最终返回 second_sell,即最大利润。

总结

  • 核心思路:通过动态规划,定义四个状态来跟踪两次买入和两次卖出的最大利润,并通过状态转移公式逐步更新,最终获取最大利润。
  • 时间复杂度:时间复杂度为 O(n),适合处理大规模数据。
  • 空间优化:由于状态之间只依赖前一天的状态,空间复杂度可以优化为 O(1),这使得算法在空间效率上达到最优。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/54916.shtml

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

相关文章

07_23 种设计模式之《桥接模式》

文章目录 一、桥接模式基础知识二、桥接模式实战应用 一、桥接模式基础知识 桥接模式定义&#xff1a;将抽象部分与它的实现部分分离&#xff0c;使它们都可以 独立地变化。桥接模式的结构图如下&#xff1a; Abstraction&#xff1a;抽象部分的接口。 RefinedAbstraction&…

Python基础之List列表用法

1、创建列表 names ["张三","李四","王五","Mary"] 2、列表分片 names[1]&#xff1a;获取数组的第2个元素。 names[1:3]&#xff1a;获取数组的第2、第3个元素。包含左侧&#xff0c;不包含右侧。 names[:3]等同于names[0:3]&…

List子接口

1.特点&#xff1a;有序&#xff0c;有下标&#xff0c;元素可以重复 2.方法&#xff1a;包含Collection中的所有方法&#xff0c;还包括自己的独有的方法&#xff08;API中查找&#xff09; 还有ListIterator&#xff08;迭代器&#xff09;&#xff0c;功能更强大。 包含更多…

llava1.5论文阅读

Improved Baselines with Visual Instruction Tuning 通过视觉指令微调增强的基线方法 论文摘要&#xff1a; 我们发现&#xff0c;LLaVA中的全连接视觉语言连接器非常强大且数据效率高。 3.3 数据和模型的scaling 受到将线性投影转变为多层感知机&#xff08;MLP&#xff0…

支付宝支付Java+uniapp支付宝小程序

JS&#xff1a; request.post(/vip/pay, {//这是自己写的java支付接口id: this.vipInfo.id,payWay: alipay-mini}).then((res) > {let success (res2) > {//前端的支付成功回调函数this.$refs.popup.close();// 支付成功刷新当前页面setTimeout(() > {this.doGetVipI…

机器学习/数据分析--用通俗语言讲解时间序列自回归(AR)模型,并用其预测天气,拟合度98%+

时间序列在回归预测的领域的重要性&#xff0c;不言而喻&#xff0c;在数学建模中使用及其频繁&#xff0c;但是你真的了解ARIMA、AR、MA么&#xff1f;ACF图你会看么&#xff1f;&#xff1f; 时间序列数据如何构造&#xff1f;&#xff1f;&#xff1f;&#xff0c;我打过不少…

读书笔记 - 虚拟化技术 - 0 QEMU/KVM概述与历史

《QEMU/KVM源码解析与应用》 - 王强 概述 虚拟化简介 虚拟化思想 David Wheeler&#xff1a;计算机科学中任何问题都可以通过增加一个中间层来解决。 虚拟化思想存在与计算机科学的各个领域。 主要思想&#xff1a;通过分层将底层的复杂&#xff0c;难用的资源虚拟抽象为简…

32单片机 低功耗模式

以下是一个基于STM32的低功耗模式示例代码&#xff0c;展示如何将STM32微控制器置于低功耗模式&#xff0c;并在特定条件下唤醒它。这个示例使用的是STM32 HAL库。 ### 示例代码&#xff1a;进入睡眠模式并使用外部中断唤醒 c #include "stm32f4xx_hal.h" // 函数声明…

笔记本电脑按住电源键强行关机,对电脑有伤害吗?

电脑卡住了&#xff0c;我们习惯性地按住电源键或者直接拔掉电源强制关机&#xff0c;但这种做法真的安全吗&#xff1f;会不会对电脑造成伤害呢&#xff1f; 其实&#xff0c;按住电源键关机和直接拔掉电源关机是不一样的。它们在硬件层面有着本质区别。 按住电源键关机 当…

网络常用配置和运维命令以及使用方法

一、网络运维常见配置及命令 以下是一些整理简单网络常用配置和运维命令 1、ifconfig/ipconfig&#xff1a;查看和配置网络接口配置。 用法&#xff1a;ifconfig 或 ipconfig2、route/netstat -r&#xff1a;查看路由表。 用法&#xff1a;route -n 或 netstat -r3、netsta…

Spring Cloud 3.x 集成eureka快速入门Demo

1.什么是eureka&#xff1f; Eureka 由 Netflix 开发&#xff0c;是一种基于REST&#xff08;Representational State Transfer&#xff09;的服务&#xff0c;用于定位服务&#xff08;服务注册与发现&#xff09;&#xff0c;以实现中间层服务的负载均衡和故障转移&#xff…

工业和自动化领域常见的通信协议

在工业和自动化领域&#xff0c;有多种常见的通信协议&#xff0c;主要用于设备间的通信、数据传输和控制。 Modbus&#xff1a; 类型&#xff1a;串行通信协议用途&#xff1a;广泛用于工业自动化设备间的通信&#xff0c;如PLC、传感器和执行器。优点&#xff1a;简单、开放且…

day01-Qt5入门

day01-Qt5入门 窗体应用 1.1 窗体基类说明 创建项目在details中编辑器提供了三个基类&#xff0c;分别是 QMainWindows、Qwidget、QDialog 1、 QMainWindow QMainWindow 类提供一个有菜单条、锚接窗口&#xff08;例如工具条&#xff09;和一个状态条的主应用 程序窗口。…

【系统架构设计师】案例专题六(8大系统架构设计之1): 信息系统架构设计考点梳理

更多内容请见: 备考系统架构设计师-核心总结目录 文章目录 一、信息系统架构基本概念二、信息系统架构1、信息系统架构分类2、信息系统常用4种架构模型3、企业信息系统的总体框架三、信息系统架构设计方法1、ADM架构开发方法2、信息化总体架构方法3、TOGAF的目标、思想和组件四…

【C语言实现TCP客户端】

Question C语言实现TCP客户端 Answer 下面是一个简单的 C 语言实现的 TCP 客户端示例。这个客户端将连接到指定的服务器地址和端口&#xff0c;并发送一条消息&#xff0c;然后接收服务器返回的消息。 #include <stdio.h> #include <stdlib.h> #include <st…

Java-基础

1. 导入模块不能纯粹的复制粘贴&#xff0c;要从new里导入&#xff0c;因为前者建立不了关联 2. 数组 String[] name{"张三","李四","王五"};int[] numsnew int[]{1,2,3};//二维String[][] names{{"张三","李四"},{"…

RT-DETR改进策略:BackBone改进|CAFormer在RT-DETR中的创新应用,显著提升目标检测性能

摘要 在目标检测领域,模型性能的提升一直是研究者和开发者们关注的重点。近期,我们尝试将CAFormer模块引入RT-DETR模型中,以替换其原有的主干网络,这一创新性的改进带来了显著的性能提升。 CAFormer,作为MetaFormer框架下的一个变体,结合了深度可分离卷积和普通自注意力…

MATLAB - 机械臂手眼标定(眼在手内) - 估计安装在机器人上的移动相机的姿态

系列文章目录 前言 本示例展示了如何为装有手眼构型摄像头的机械臂或机械手执行和验证手眼校准。 一、概述 执行手眼校准有助于操作配备末端执行器&#xff08;简称 “手”&#xff09;的机械臂&#xff0c;该末端执行器依赖于摄像头提供的视觉数据。一旦完成了眼在手外的校准&…

LabVIEW 成绩统计系统

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…

【算法】- 查询 -平衡二叉树

文章目录 前言一、平衡二叉树思想二、如何进行平衡调整2.1如何实现左平衡&#xff08;L_Balance&#xff09; 三、平衡二叉树总结 前言 编译语言&#xff1a;C 编译器&#xff1a;VS2022 一、平衡二叉树思想 在上节我们讲到了二叉排序树&#xff0c;当我们在使用二叉排序树时&a…