代码随想录训练营 Day41打卡 动态规划 part08 121. 买卖股票的最佳时机 122. 买卖股票的最佳时机II 123. 买卖股票的最佳时机III

代码随想录训练营 Day41打卡 动态规划 part08

一、力扣121. 买卖股票的最佳时机

给定一个数组 prices ,它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。
你只能选择 某一天 买入这只股票,并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所能获取的最大利润。
返回你可以从这笔交易中获取的最大利润。如果你不能获取任何利润,返回 0 。
示例
输入:[7,1,5,3,6,4]
输出:5
解释:在第 2 天(股票价格 = 1)的时候买入,在第 5 天(股票价格 = 6)的时候卖出,最大利润 = 6-1 = 5 。
注意利润不能是 7-1 = 6, 因为卖出价格需要大于买入价格;同时,你不能在买入前卖出股票。

在这个问题中,我们需要计算在给定的股票价格数组 prices 中,最多只能进行一次交易(即一次买入和一次卖出)的情况下,能够获得的最大利润。为了实现这一点,我们使用动态规划的方法来模拟整个买卖过程。

我们定义两个状态:

  • dp[i][0] 表示第 i 天持有股票时能够获得的最大现金金额。
  • dp[i][1] 表示第 i 天不持有股票时能够获得的最大现金金额。

注意,“持有股票”不一定表示当天买入股票,也可能是之前买入的,并且持续持有到当天。同样地,“不持有股票”可以是当天卖出股票,或者之前已经卖出并且保持不持有状态。

持有股票时 (dp[i][0]):

如果第 i-1 天已经持有股票,那么保持现状:dp[i - 1][0]。
如果第 i 天买入股票,那么所得现金为 -prices[i]。
公式:dp[i][0] = max(dp[i - 1][0], -prices[i])

不持有股票时 (dp[i][1]):

如果第 i-1 天已经不持有股票,那么保持现状:dp[i - 1][1]。
如果第 i 天卖出股票,那么所得现金为 prices[i] + dp[i - 1][0]。
公式:dp[i][1] = max(dp[i - 1][1], prices[i] + dp[i - 1][0])
以示例输入:[7,1,5,3,6,4]为例,dp数组状态如下:
在这里插入图片描述
dp[5][1]就是最终结果。

为什么不是dp[5][0]呢?

因为本题中不持有股票状态所得金钱一定比持有股票状态得到的多!

代码实现

class Solution:def maxProfit(self, prices: List[int]) -> int:length = len(prices)if length == 0:return 0# 初始化 dp 数组,大小为 [length][2]# dp[i][0] 表示第 i 天持有股票时的最大现金# dp[i][1] 表示第 i 天不持有股票时的最大现金dp = [[0] * 2 for _ in range(length)]# 初始化第一天的状态dp[0][0] = -prices[0]  # 第 0 天买入股票dp[0][1] = 0  # 第 0 天不持有股票# 动态规划,计算每一天的状态for i in range(1, length):# 第 i 天持有股票的最大现金dp[i][0] = max(dp[i - 1][0], -prices[i])# 第 i 天不持有股票的最大现金dp[i][1] = max(dp[i - 1][1], prices[i] + dp[i - 1][0])# 最终结果是最后一天不持有股票时的最大现金return dp[-1][1]

力扣题目链接
题目文章讲解
题目视频讲解

二、力扣122. 买卖股票的最佳时机II

给你一个整数数组 prices ,其中 prices[i] 表示某支股票第 i 天的价格。
在每一天,你可以决定是否购买和/或出售股票。你在任何时候 最多 只能持有 一股 股票。你也可以先购买,然后在 同一天 出售。
返回 你能获得的 最大 利润 。
示例
输入:prices = [7,1,5,3,6,4]
输出:7
解释:在第 2 天(股票价格 = 1)的时候买入,在第 3 天(股票价格 = 5)的时候卖出, 这笔交易所能获得利润 = 5 - 1 = 4。
随后,在第 4 天(股票价格 = 3)的时候买入,在第 5 天(股票价格 = 6)的时候卖出, 这笔交易所能获得利润 = 6 - 3 = 3。
最大总利润为 4 + 3 = 7 。

在这个问题中,我们需要找到一个策略,使得通过多次交易(买入和卖出股票)能够获得的最大利润。与之前只能进行一次交易的情况不同,这里允许在多个不同的时间点进行多次交易,每次交易只能持有一股股票。

我们仍然使用动态规划来解决这个问题。我们定义两个状态:

dp[i][0] 表示第 i 天持有股票时能够获得的最大现金金额。
dp[i][1] 表示第 i 天不持有股票时能够获得的最大现金金额。

与只能进行一次交易不同,这里第 i 天买入股票的现金金额可以是前一天不持有股票时的现金减去今天的股票价格,即 dp[i-1][1] - prices[i]。

持有股票时 (dp[i][0]):

如果第 i-1 天已经持有股票,那么保持现状:dp[i-1][0]。
如果第 i 天买入股票,那么所得现金为前一天不持有股票的现金减去今天的股票价格:dp[i-1][1] - prices[i]。
公式:dp[i][0] = max(dp[i-1][0], dp[i-1][1] - prices[i])

不持有股票时 (dp[i][1]):

如果第 i-1 天已经不持有股票,那么保持现状:dp[i-1][1]。
如果第 i 天卖出股票,那么所得现金为今天的股票价格加上前一天持有股票的现金:dp[i-1][0] + prices[i]。
公式:dp[i][1] = max(dp[i-1][1], dp[i-1][0] + prices[i])

代码实现

class Solution:def maxProfit(self, prices: List[int]) -> int:length = len(prices)# 特殊情况处理if length == 0:return 0# 初始化 dp 数组,大小为 [length][2]# dp[i][0] 表示第 i 天持有股票时的最大现金# dp[i][1] 表示第 i 天不持有股票时的最大现金dp = [[0] * 2 for _ in range(length)]# 初始化第一天的状态dp[0][0] = -prices[0]  # 第 0 天买入股票dp[0][1] = 0  # 第 0 天不持有股票# 动态规划,计算每一天的状态for i in range(1, length):# 第 i 天持有股票的最大现金dp[i][0] = max(dp[i-1][0], dp[i-1][1] - prices[i])# 第 i 天不持有股票的最大现金dp[i][1] = max(dp[i-1][1], dp[i-1][0] + prices[i])# 最终结果是最后一天不持有股票时的最大现金return dp[-1][1]

力扣题目链接
题目文章讲解
题目视频讲解

二、力扣123. 买卖股票的最佳时机III

给定一个数组,它的第 i 个元素是一支给定的股票在第 i 天的价格。
设计一个算法来计算你所能获取的最大利润。你最多可以完成 两笔 交易。
注意:你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。
示例
输入:prices = [3,3,5,0,0,3,1,4]
输出:6
解释:在第 4 天(股票价格 = 0)的时候买入,在第 6 天(股票价格 = 3)的时候卖出,这笔交易所能获得利润 = 3-0 = 3 。
随后,在第 7 天(股票价格 = 1)的时候买入,在第 8 天 (股票价格 = 4)的时候卖出,这笔交易所能获得利润 = 4-1 = 3 。

在这个问题中,我们需要找到一个策略,通过最多两次交易(即最多买入两次和卖出两次)来获得最大利润。每次交易只能持有一股股票,因此有五种状态:

dp[i][0]:表示第 i 天没有任何操作或不进行任何交易时的最大利润。
dp[i][1]:表示第 i 天进行了第一次买入操作后的最大利润。
dp[i][2]:表示第 i 天进行了第一次卖出操作后的最大利润。
dp[i][3]:表示第 i 天进行了第二次买入操作后的最大利润。
dp[i][4]:表示第 i 天进行了第二次卖出操作后的最大利润。

dp[i][1]: 第 i 天持有股票(第一次买入)的最大利润:
如果第 i 天买入股票:dp[i-1][0] - prices[i]。
如果第 i 天不买入股票:dp[i-1][1]。
公式:dp[i][1] = max(dp[i-1][1], dp[i-1][0] - prices[i])

dp[i][2]: 第 i 天不持有股票(第一次卖出)的最大利润:
如果第 i 天卖出股票:dp[i-1][1] + prices[i]。
如果第 i 天不卖出股票:dp[i-1][2]。
公式:dp[i][2] = max(dp[i-1][2], dp[i-1][1] + prices[i])

dp[i][3]: 第 i 天持有股票(第二次买入)的最大利润:
如果第 i 天买入股票:dp[i-1][2] - prices[i]。
如果第 i 天不买入股票:dp[i-1][3]。
公式:dp[i][3] = max(dp[i-1][3], dp[i-1][2] - prices[i])

dp[i][4]: 第 i 天不持有股票(第二次卖出)的最大利润:
如果第 i 天卖出股票:dp[i-1][3] + prices[i]。
如果第 i 天不卖出股票:dp[i-1][4]。
公式:dp[i][4] = max(dp[i-1][4], dp[i-1][3] + prices[i])

dp数组初始化

dp[0][0] = 0:表示第 0 天不进行任何操作,利润为 0。
dp[0][1] = -prices[0]:表示第 0 天第一次买入股票后的利润,即减去第 0 天的股票价格。
dp[0][2] = 0:表示第 0 天第一次卖出股票后的利润,因为不能卖出不存在的股票,所以利润为 0。
dp[0][3] = -prices[0]:表示第 0 天第二次买入股票后的利润,即再减去第 0 天的股票价格。
dp[0][4] = 0:表示第 0 天第二次卖出股票后的利润,同理也为 0。

代码实现

class Solution:def maxProfit(self, prices: List[int]) -> int:# 特殊情况处理,如果没有股票价格数据,返回 0if len(prices) == 0:return 0# 初始化 dp 数组,大小为 [len(prices)][5]dp = [[0] * 5 for _ in range(len(prices))]# 第 0 天的状态初始化dp[0][0] = 0  # 不操作dp[0][1] = -prices[0]  # 第一次买入dp[0][2] = 0  # 第一次卖出dp[0][3] = -prices[0]  # 第二次买入dp[0][4] = 0  # 第二次卖出# 动态规划,遍历每一天的状态for i in range(1, len(prices)):# 第 i 天不操作的状态,延续前一天的状态dp[i][0] = dp[i-1][0]# 第 i 天第一次买入股票的状态dp[i][1] = max(dp[i-1][1], dp[i-1][0] - prices[i])# 第 i 天第一次卖出股票的状态dp[i][2] = max(dp[i-1][2], dp[i-1][1] + prices[i])# 第 i 天第二次买入股票的状态dp[i][3] = max(dp[i-1][3], dp[i-1][2] - prices[i])# 第 i 天第二次卖出股票的状态dp[i][4] = max(dp[i-1][4], dp[i-1][3] + prices[i])# 最终返回的结果是最后一天,第二次卖出股票后的最大利润return dp[-1][4]

力扣题目链接
题目文章讲解
题目视频讲解

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

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

相关文章

反事实推理(Counterfactual Reasoning):探索未知与决策的桥梁

反事实推理(Counterfactual Reasoning):探索未知与决策的桥梁 反事实推理(Counterfactual Reasoning)是一种思维方式,它试图回答“如果……会怎样?”的问题。简单来说,反事实推理是…

中国料箱穿梭车玩家TOP榜单

导语 大家好,我是社长,老K。专注分享智能制造和智能仓储物流等内容。 新书《智能物流系统构成与技术实践》人俱乐部 料箱穿梭车前景 随着全球智慧物流建设的加速推进,智能仓储物流成为未来发展的重要趋势。在此背景下,料箱穿梭车作…

cuda,torch,paddle向下兼容

1、第一次配置yolov9模型时,使用的cuda的版本是11.6,torch和torchvision都是对应版本的 使用的tensorrt版本8.6,可以正常跑yolov9 其它不动,直接将cuda版本换为cuda11.7,依然可以正常运行 2、paddleseg paddle同样安…

carla unreal engine源码:如何创建radar可视化探测锥

文章目录 前言一、C实现方法1、DrawDebugCone函数2、carla工程修改3、make launch4、探测锥验证 二、蓝图实现方法1、创建并打开蓝图2、打开蓝图事件图表3、绘制蓝图事件4、编译再运行 前言 1、在自动驾驶仿真调试以及测试过程中,我们经常会用到雷达的探测锥&#…

OpenCV小练习:身份证号码识别

目标:针对一张身份证照片,把身份证号码识别出来(转成数字或字符串)。 实现思路:需要将目标拆分成两个子任务:(1) 把身份证号码区域从整张图片中检测/裁剪出来;(2) 将图片中的数字转化成文字。第…

快速学习go-zero

go的web框架有很多,目前go的社区大家对于框架的态度也不尽相同,有些轻量级的框架,但是也就代表整合第三方中间件就需要自己根据客户端进行封装,比如gingorm,也有些功能完全但是被认为丢失了go本身轻量设计的初衷, 比如goframe,而同样的微服务有很多框架,国内比较出门的就是go-z…

rockyliunx 救援模式下禁用docker

目录地址 /usr/lib/systemd/system/docker.service 进入系统界面: 选择系统 按E 按e出现 如下界面,找到 quite 后面添加 init/bin/bash 按 ctrl x 保存 后,到如下界面 加载文件系统为读写 输入命令 mount -o remount, rw / 修改docer.s…

docker的安装+docker镜像的基本操作

一.docker的介绍 1、Docker 是什么? Docker 是⼀个开源的应⽤容器引擎,可以实现虚拟化,完全采⽤“沙 盒”机制,容器之间不会存在任何接⼝。 Docker 通过 Linux Container(容器)技术将任意…

【线程池】

什么是线程池? 线程池是一个可以复用线程的技术。简单来说,线程池是一种基于池化技术的思想来管理线程的技术,旨在减少线程的创建和销毁次数,提高系统的响应速度和吞吐量。它预先创建了一定数量的线程,并将这些线程放…

Java中的定时器(Timer)

目录 一、什么是定时器? 二、标准库中的定时器 三、实现自定义定时器 一、什么是定时器? 定时器就像一个"闹钟",当它到达设定的时间后,就会执行预定的代码。 例如,我们在TCP的超时重传机制中讲过,如果服务器在规定…

DNS劫持问题

目录 DNS劫持概述 定义 图示 ​编辑图示说明 DNS劫持的原理 1. DNS请求与响应过程 图示 ​编辑2. 劫持发生点 本地劫持 路由器劫持 中间人攻击 图示 ​编辑图示说明 DNS劫持的影响 1. 对个人用户的影响 图示 ​编辑图示说明 2. 对企业的影响 图示 ​编辑图示…

0828作业+梳理

一、作业 代码&#xff1a; #include <iostream>using namespace std;using datatype int; //类型重命名 #define MAX 2 //宏定义 //结构体定义 struct Sqlist { private:datatype *data; //顺序表数组int size 0; //数组大小int len 0; …

Ubuntu 16.04下Firefox版本更新

​ 大家好&#xff0c;我是程序员小羊&#xff01; 前言&#xff1a; 在Ubuntu 16.04上更新Firefox的过程可能涉及多个步骤&#xff0c;具体取决于你的需求&#xff0c;比如是要安装一个稳定版本&#xff0c;还是需要使用最新的开发者版本或beta版本。下面我将详细介绍如何在Ub…

Python酷库之旅-第三方库Pandas(105)

目录 一、用法精讲 456、pandas.DataFrame.rdiv方法 456-1、语法 456-2、参数 456-3、功能 456-4、返回值 456-5、说明 456-6、用法 456-6-1、数据准备 456-6-2、代码示例 456-6-3、结果输出 457、pandas.DataFrame.rtruediv方法 457-1、语法 457-2、参数 457-3…

搭建面向切面编程项目

此项目在整合Mybatis基础上修改&#xff0c;可参考主页的整合Mybatis文章 注解版本 第一步 引入maven坐标 <!-- 切面编程所需jar包--><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId>…

学生管理系统升级(登录注册 + 关联学生管理系统)

新增需求 这是在昨天的基础初代版本上面新增一个登录注册忘记密码的功能 需求分析 注册 登录 忘记密码 user类代码呈现 package StudentSystem;public class User {private String username;private String password;private String personID;private String phoneNumber;pu…

PHP同城派送多区域运营配送小程序源码

&#x1f69a;&#x1f4a8;「同城派送多区域运营小程序」——让每一份需求快速触达&#xff01;&#x1f308;&#x1f680; &#x1f525; 开篇燃爆&#xff1a;同城生活新风尚&#xff0c;一键速达不是梦&#xff01; Hey小伙伴们&#xff0c;你还在为找不到合适的同城服务…

推荐并整理一波vscode插件(哪些内置了,哪些好用)

文章目录 背景现在还在用的&#xff08;21款&#xff09;Chinese(Simplified)简体中文Chinese LoremLorem ipsumCode Runner&#xff08;很推荐&#xff09;Codeium: AI Coding Autocomplete&#xff08;推荐&#xff09;Draw.io IntegrationESLintHighlight Matching TagJavaS…

甄选范文“论软件系统建模方法及其应用”,软考高级论文,系统架构设计师论文

论文真题 软件系统建模(Software System Modeling)是软件开发中的重要环节,通过构建软件系统模型可以帮助系统开发人员理解系统、抽取业务过程和管理系统的复杂性,也可以方便各类人员之间的交流。软件系统建模是在系统需求分析和系统实现之间架起的一座桥梁,系统开发人员…