当前位置: 首页 > news >正文

经典动态规划问题:含手续费的股票买卖【从 O(n) 到 O(1) 的优化解析】

题目理解

我们要在给定的股票价格数组 prices 中进行买卖操作,并尽可能多次交易以获取最大利润。每次交易都需要支付一定的手续费 fee,因此我们必须考虑如何通过合适的交易策略最大化利润。

在本题中,每一天可以选择:

  1. 不进行任何操作。
  2. 买入股票。
  3. 卖出股票(前提是已经持有股票)。

题目链接:714. 买卖股票的最佳时机含手续费 - 力扣(LeetCode)

动态规划思路

我们可以使用动态规划(DP)来解决这个问题。在动态规划中,我们定义两种状态:

  1. 持有股票状态(持仓)hold[i] 表示第 i 天结束时持有股票时的最大利润。
  2. 不持有股票状态(空仓)cash[i] 表示第 i 天结束时不持有股票时的最大利润。
状态转移方程
  • 持有股票的状态
    我们有两种可能:

    1. 我们在第 i 天之前已经持有股票,那么 hold[i] 就和 hold[i-1] 相同。
    2. 我们在第 i 天买入股票,那么需要从 cash[i-1](前一天不持有股票的最大利润)中减去 prices[i](当天股票价格)。

      因此,持有股票状态的转移方程为:
      h o l d [ i ] = max ⁡ ( h o l d [ i − 1 ] , c a s h [ i − 1 ] − p r i c e s [ i ] ) hold[i] = \max(hold[i-1], cash[i-1] - prices[i]) hold[i]=max(hold[i1],cash[i1]prices[i])
  • 不持有股票的状态
    我们有两种可能:

    1. 我们在第 i 天之前已经卖出了股票,那么 cash[i] 就和 cash[i-1] 相同。
    2. 我们在第 i 天卖出股票,此时需要加上 prices[i] 的收入并扣除手续费 fee

      因此,不持有股票状态的转移方程为:
      c a s h [ i ] = max ⁡ ( c a s h [ i − 1 ] , h o l d [ i − 1 ] + p r i c e s [ i ] − f e e ) cash[i] = \max(cash[i-1], hold[i-1] + prices[i] - fee) cash[i]=max(cash[i1],hold[i1]+prices[i]fee)
初始状态
  • hold[0] = -prices[0]:第0天如果买入股票,我们的利润就是负的第0天的股票价格。
  • cash[0] = 0:第0天如果不买股票,利润为0。
最终结果

我们最终需要的是在最后一天结束时,不持有股票时的最大利润,即 cash[n-1],其中 nprices 的长度。

C++ 实现

#include <vector>
#include <algorithm>
using namespace std;int maxProfit(vector<int>& prices, int fee) {int n = prices.size();vector<int> hold(n), cash(n);hold[0] = -prices[0]; // 第 0 天持有股票cash[0] = 0;          // 第 0 天不持有股票for (int i = 1; i < n; ++i) {cash[i] = max(cash[i-1], hold[i-1] + prices[i] - fee); // 不持有股票hold[i] = max(hold[i-1], cash[i-1] - prices[i]);       // 持有股票}return cash[n-1];  // 最后一天不持有股票的最大利润
}

优化思路

这个基本解法需要两个数组 holdcash,分别存储持有和不持有股票时的利润值。这会占用 O(n) 的空间。而实际上,在计算第 i 天的状态时,只依赖于 i-1 天的状态,因此我们可以使用两个变量代替数组,优化空间复杂度到 O(1)

优化后的实现

#include <vector>
#include <algorithm>
using namespace std;int maxProfit(vector<int>& prices, int fee) {int n = prices.size();int hold = -prices[0]; // 持有股票时的最大利润int cash = 0;          // 不持有股票时的最大利润for (int i = 1; i < n; ++i) {cash = max(cash, hold + prices[i] - fee); // 更新不持有股票的最大利润hold = max(hold, cash - prices[i]);       // 更新持有股票的最大利润}return cash;  // 最后一天不持有股票的最大利润
}

解释及示例

示例 1

输入:prices = [1, 3, 2, 8, 4, 9], fee = 2
输出:8

过程:

  1. 第 0 天:买入股票,hold = -1cash = 0
  2. 第 1 天:卖出股票,cash = max(0, -1 + 3 - 2) = 0,保持持有状态,hold = -1
  3. 第 2 天:保持持有状态,cash = 0hold = max(-1, 0 - 2) = -1
  4. 第 3 天:卖出股票,cash = max(0, -1 + 8 - 2) = 5hold = max(-1, 5 - 8) = -1
  5. 第 4 天:买入股票,cash = 5hold = max(-1, 5 - 4) = 1
  6. 第 5 天:卖出股票,cash = max(5, 1 + 9 - 2) = 8hold = 1

最终结果:cash = 8

示例 2

输入:prices = [1, 3, 7, 5, 10, 3], fee = 3
输出:6

  1. 第 0 天:买入股票,持有股票的利润为 hold = -1,不持有股票的利润为 cash = 0
  2. 第 1 天:卖出股票后利润为 cash = max(0, -1 + 3 - 3) = 0,持有状态 hold = max(-1, 0 - 3) = -1
  3. 第 2 天:卖出股票后利润为 cash = max(0, -1 + 7 - 3) = 3,持有状态 hold = max(-1, 3 - 7) = -1
  4. 第 3 天:保持不持有状态 cash = max(3, -1 + 5 - 3) = 3,持有状态 hold = max(-1, 3 - 5) = -1
  5. 第 4 天:卖出股票后利润为 cash = max(3, -1 + 10 - 3) = 6,持有状态 hold = max(-1, 6 - 10) = -1
  6. 第 5 天:保持不持有状态 cash = max(6, -1 + 3 - 3) = 6

最终利润为 6。

关键点总结

  1. 最优子结构:第 i 天的状态只取决于第 i-1 天的状态。

  2. 状态转移方程

    • 持有状态:hold[i] = max(hold[i-1], cash[i-1] - prices[i])
    • 不持有状态:cash[i] = max(cash[i-1], hold[i-1] + prices[i] - fee)
  3. 空间优化:我们只需要两个变量 holdcash,可以将空间复杂度从 O(n) 优化到 O(1)

相关文章:

经典动态规划问题:含手续费的股票买卖【从 O(n) 到 O(1) 的优化解析】

题目理解 我们要在给定的股票价格数组 prices 中进行买卖操作&#xff0c;并尽可能多次交易以获取最大利润。每次交易都需要支付一定的手续费 fee&#xff0c;因此我们必须考虑如何通过合适的交易策略最大化利润。 在本题中&#xff0c;每一天可以选择&#xff1a; 不进行任…...

Python画笔案例-088 绘制 滚动的汉字

1、绘制 滚动的汉字 通过 python 的turtle 库绘制 滚动的汉字,如下图: 2、实现代码 绘制 滚动的汉字,以下为实现代码: """滚动的汉字.py """ import time from turtle import * from write_patch import *width,height...

Redis 5.0 安装配置(Windows)

Redis 5.0之后支持Redis Stream等功能 下载地址&#xff1a;Releases tporadowski/redis GitHub 点击运行redis-server.exe 此外&#xff1a;Redis 6.0及以后版本目前都没有Windows版...

金融行业:办公安全防护专属攻略

在中国金融市场快速迈向数字化转型的进程中&#xff0c;数据安全与隐私保护成为了不容忽视的关键议题。面对不断升级的网络威胁和日益严格的监管要求&#xff0c;构建一个既能支持创新又能提供坚实防护的信息安全体系变得尤为重要。亿格云不断深耕办公安全领域&#xff0c;为金…...

python如何基于numpy pandas完成复杂的数据分析操作?

数据分析是现代数据科学的重要组成部分,Python作为一种强大的编程语言,提供了许多库来简化数据分析过程。 其中,NumPy和Pandas是两个最常用的库。NumPy主要用于数值计算,而Pandas则提供了强大的数据结构和数据分析工具。 本文将深入探讨如何利用这两个库进行复杂的数据分…...

Linux中定时任务调度工具——crontab

1.简介 crontab是Unix和类Unix操作系统&#xff08;如Linux&#xff09;中用于定时任务调度的工具。其名称来源于“cron”这个守护进程&#xff0c;它负责周期性地执行任务&#xff0c;并且“tab”表示这个工具的配置文件。用户可以通过配置crontab文件来设定定时任务&#xf…...

思维+差分,CF 1884C - Medium Design

目录 一、题目 1、题目描述 2、输入输出 2.1输入 2.2输出 3、原题链接 二、解题报告 1、思路分析 2、复杂度 3、代码详解 一、题目 1、题目描述 2、输入输出 2.1输入 2.2输出 3、原题链接 1884C - Medium Design 二、解题报告 1、思路分析 考虑 最大值 和 最小值…...

简单介绍冯诺依曼体系

现代的计算机, 大多遵守冯诺依曼体系结构 CPU中央处理器&#xff1a;进行算术运算和逻辑判断。存储器&#xff1a;分为外存和内存&#xff0c;用于存储数据&#xff08;使用二进制方式存储&#xff09;。输入设备&#xff1a;用户给计算机发号施令。输出设备&#xff1a;计算机…...

kernel32.dll下载地址:如何安全地恢复系统文件

关于从网络上寻找kernel32.dll的下载地址&#xff0c;这通常不是一个安全的做法&#xff0c;而且可能涉及到多种风险。kernel32.dll是Windows操作系统的核心组件之一&#xff0c;负责内存管理、进程和线程管理以及其他关键系统功能。因为kernel32.dll是系统的基础文件&#xff…...

【高等数学】多元微分学 (一)

偏导数 偏导数定义 如果二元函数 f f f 在 x 0 , y 0 x_0,y_0 x0​,y0​ 的某邻域有定义, 且下述极限存在 lim ⁡ Δ x → 0 f ( x 0 Δ x , y 0 ) − f ( x 0 , y 0 ) Δ x \lim_{\Delta x\to 0} \frac{f(x_0\Delta x,y_0)-f(x_0,y_0)}{\Delta x} Δx→0lim​Δxf(x0​Δ…...

Python爬取京东商品信息,详细讲解,手把手教学(附源码)

Python 爬虫爬取京东商品信息 下面我将逐一解释每一部分的代码 导入库 from selenium import webdriver from selenium.webdriver.edge.service import Service from selenium.webdriver.edge.options import Options import time import random import csv from selenium.c…...

大家有没有了解过TLKS-PLGS这款接地电阻在线监测装置?它在电力系统中能发挥什么作用呢?

接地电阻在线监测仪&#xff08;输电铁塔接地电阻监测装置、变电站接地电阻监测装置、三极法接地网电阻监测装置&#xff09;在电力系统中发挥着至关重要的作用&#xff0c;具体来说&#xff0c;有以下几个方面&#xff1a; 一、实时监测预警。该装置采用激励脉冲技术&#xf…...

Shell中的函数

目录 一、系统函数 &#xff08;一&#xff09;前言 &#xff08;二&#xff09;常用的函数 basename [string/pathname] [suffix] 二、自定义函数 &#xff08;一&#xff09;语法 &#xff08;二&#xff09;脚本例子 三、函数实际案例 过程中的报错&#xff1a; …...

通过IP地址或者主机名添加打印机20241023

文印室打印机连接方式20241023 Win键盘搜索打印机和扫描仪点击添加打印机或扫描仪&#xff0c;等候片刻点击“我需要的打印机不在列表中”添加打印机&#xff0c;选择使用IP地址或主机名添加打印机点击下一步&#xff0c;设备类型选择自动检测输入主机名&#xff1a;即打印机有…...

基于SpringBoot+Vue智慧养老关爱系统【提供源码+答辩PPT+参考文档+项目部署】

&#x1f4a5; 这两年毕业设计和毕业答辩的要求和难度不断提升&#xff0c;传统的JavaWeb项目缺少创新和亮点&#xff0c;往往达不到毕业答辩的要求&#xff01; ❗如何解决这类问题&#xff1f; 让我们能够顺利通过毕业&#xff0c;我也一直在不断思考、努力、精进。通过2024年…...

新手教学系列——利用短效代理快速搭建代理池

引言 在进行高并发数据抓取时,很多人都会遇到频繁IP被封的问题。要解决这个问题,代理池的搭建就成了关键。通过频繁更换代理IP,我们可以绕过网站的反爬机制,提升抓取效率。然而,很多初学者可能会觉得构建一个健壮的代理池颇为复杂,尤其是需要快速切换的短效代理池。在这…...

实体与DTO如何转换

下面是一些常用的转换库&#xff1a; Dozer 该项目目前不活跃&#xff0c;并且很可能在未来被弃用。 ModelMapper 一个智能对象映射库&#xff0c;可自动将对象相互映射。它采用基于约定的方法&#xff0c;同时提供简单、重构安全的应用程序接口&#xff08;API&#xff09;来…...

Docker 安装Postgres和PostGIS,并制作镜像

1. 查找postgres和postgis现有的镜像和版本号 镜像搜索网站&#xff1a;https://docker.aityp.com/ 测试使用的是postgres:15.4 和 postgis:15-3.4 2、镜像拉取 docker pull postgres:15.4docker pull postgis/postgis:15-3.4镜像下载完成&#xff0c;docker images 查看如…...

ES6:let和const命令解读以及变量的解构赋值

有时候&#xff0c;我们需要的不是答案&#xff0c;而是一双倾听的耳朵 文章目录 let和const命令变量的解构赋值 let和const命令 let和const命令都是声明变量的关键字&#xff0c;类同varlet特点 用来声明变量&#xff0c;不能再次定义&#xff0c;但是值可以改变存在块级作用…...

java-collection集合整理0.9.4

java-集合整理0.9.0 基本结构基本概念实例化举例遍历获取指定值 2024年10月17日09:43:16–0.9.0 2024年10月18日11:00:59—0.9.4 基本结构 Collection 是最顶级的接口。分为 List 和 Set 两大类。List 分为&#xff1a;ArrayList、LinkedList、Vector。Set 分为&#xff1a;Ha…...

9.5 点云采样——拓扑采样

图9-5-1 PointNet++中的邻域特征聚合的拓扑采样过程 拓扑/图结构采样的核心思想是“基于点云的局部拓扑关系(如K近邻、聚类)”进行采样,通过构建点云的拓扑图或聚类结构,选取每个局部区域的代表点,实现“局部保特征、全局均匀”的采样效果。 (1)出处 &n...

W4A4量化技术:OSC框架如何实现高效LLM部署

1. OSC框架&#xff1a;硬件高效的W4A4量化革命在大型语言模型(LLM)部署领域&#xff0c;4-bit量化(W4A4)正成为突破算力瓶颈的关键技术。传统8-bit量化虽已成熟&#xff0c;但当我们将精度压缩至4-bit时&#xff0c;激活张量中的异常值(Outliers)会像"黑洞"般吞噬有…...

Yaskawa JACP-317800输入输出模块

安川JACP-317800是一款高性能逻辑输入输出模块&#xff0c;隶属于安川CP-317系列PLC系统&#xff0c;专为工业自动化领域的数字信号采集与控制而设计。产品特点&#xff1a;产品类型为逻辑输入输出模块&#xff0c;作为PLC与现场设备之间的信号接口模块重量仅0.3公斤&#xff0…...

基于Ollama与Stable Diffusion的Discord AI机器人本地部署指南

1. 项目概述&#xff1a;一个能聊能画的Discord AI机器人 最近在折腾一个挺有意思的玩意儿&#xff1a;一个部署在自己电脑上的Discord机器人&#xff0c;它不仅能像ChatGPT一样跟你聊天&#xff0c;还能根据你的描述生成图片。这个项目的核心&#xff0c;是把两个当下很火的开…...

出境游网络解决方案大揭秘:eSIM 与非 eSIM 谁更胜一筹?

海外 eSIM 怎么买&#xff1f;线上直接下单就行最近几年&#xff0c;出境游再度火热起来。每次出发前&#xff0c;搞定酒店和大交通后&#xff0c;还得买手机卡。理论上&#xff0c;可带三大运营商的卡出境并开国际漫游&#xff0c;但买当地号卡和套餐更划算。去年 iPhone Air …...

告别按钮!用Qt实现STM32小车的键盘与手柄控制方案(附串口通信源码)

超越按钮控制&#xff1a;Qt框架下STM32小车的键盘与手柄交互方案 在嵌入式开发领域&#xff0c;人机交互体验往往被忽视&#xff0c;而实际上它直接影响着用户的操作效率和舒适度。对于STM32遥控小车这类需要实时操控的项目&#xff0c;传统的按钮点击方式存在明显局限——操作…...

使用Taotoken CLI工具一键配置多开发环境下的API访问密钥

&#x1f680; 告别海外账号与网络限制&#xff01;稳定直连全球优质大模型&#xff0c;限时半价接入中。 &#x1f449; 点击领取海量免费额度 使用Taotoken CLI工具一键配置多开发环境下的API访问密钥 在团队协作或个人多设备开发场景中&#xff0c;为不同的AI开发工具&…...

解密智能图片分层:掌握Layerdivider提升设计效率的实战指南

解密智能图片分层&#xff1a;掌握Layerdivider提升设计效率的实战指南 【免费下载链接】layerdivider A tool to divide a single illustration into a layered structure. 项目地址: https://gitcode.com/gh_mirrors/la/layerdivider 在数字创意领域&#xff0c;我们常…...

新手也能看懂的CrackMe逆向实战:从查壳到用OD改跳转,一步步带你破解

新手也能看懂的CrackMe逆向实战&#xff1a;从查壳到用OD改跳转&#xff0c;一步步带你破解 逆向工程就像拆解一个神秘的黑匣子&#xff0c;而CrackMe则是专门为练习破解设计的"玩具程序"。记得我第一次接触CrackMe时&#xff0c;面对满屏的汇编代码完全不知所措。本…...

LyricsX:一站式macOS歌词同步解决方案,让音乐体验更智能

LyricsX&#xff1a;一站式macOS歌词同步解决方案&#xff0c;让音乐体验更智能 【免费下载链接】LyricsX &#x1f3b6; Ultimate lyrics app for macOS. 项目地址: https://gitcode.com/gh_mirrors/ly/LyricsX LyricsX是macOS平台上功能最全面的歌词同步工具&#xff…...