代码随想录刷题day50|股票买卖的最佳时机III(至多2次买入和卖出)股票买卖的最佳时机IV(至多K次买入和卖出)

本文主要是介绍代码随想录刷题day50|股票买卖的最佳时机III(至多2次买入和卖出)股票买卖的最佳时机IV(至多K次买入和卖出),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • day50学习内容
  • 一、股票买卖的最佳时机III--至多2次买入和卖出
    • 1.1、动态规划五部曲
      • 1.1.1、 确定dp数组(dp table)以及下标的含义
      • 1.1.2、确定递推公式
      • 1.1.3、 dp数组如何初始化
      • 1.1.4、确定遍历顺序
    • 1.2、代码
      • 1.2.1、如何理解这段代码
  • 二、股票买卖的最佳时机IV--至多K次买入和卖出
    • 2.1、动态规划五部曲
      • 2.1.1、 确定dp数组(dp table)以及下标的含义
      • 2.1.2、确定递推公式
      • 2.1.3、 dp数组如何初始化
      • 2.1.4、确定遍历顺序
      • 2.1.5、输出结果
    • 2.2、代码
      • 2.2.1、代码抄的,太难了
  • 总结
    • 1.感想
    • 2.思维导图


day50学习内容

day50主要内容

  • 股票买卖的最佳时机III–至多2次买入和卖出
  • 股票买卖的最佳时机IV–至多K次买入和卖出

声明
本文思路和文字,引用自《代码随想录》


一、股票买卖的最佳时机III–至多2次买入和卖出

123.原题链接

1.1、动态规划五部曲

1.1.1、 确定dp数组(dp table)以及下标的含义

dp[i][j] 代表的是在第 i 天结束时,根据不同的交易状态(即 j),所能达到的最大利润。这个动态规划表通过不断更新这些状态,帮助我们找到在规定的交易次数内(本题为两次)能获得的最大利润。

1.1.2、确定递推公式

对于每一天 i 和每个状态 j,如下计算 dp[i][j]

  1. 没有操作 (j = 0): 这个状态实际上不参与计算,因为我们总是从第一次买入开始计算,但如果要表示它的话,可以认为它是 dp[i][0] = 0

  2. 第一次买入 (j = 1):

dp[i][1]=max(dp[i−1][1],−prices[i])

这意味着第一次买入的最大利润是上一天已经买入的最大利润和今天买入(当前价格的负值,因为我们支付了这么多钱)之间的较大者。

  1. 第一次卖出 (j = 2):
dp[i][2]=max(dp[i−1][2],dp[i−1][1]+prices[i])

第一次卖出的最大利润是上一天卖出的最大利润和今天卖出(即上一天买入的最大利润加上今天的价格)之间的较大者。

  1. 第二次买入 (j = 3):
dp[i][3]=max(dp[i−1][3],dp[i−1][2]−prices[i])

第二次买入的最大利润是上一天第二次买入的最大利润和上一天第一次卖出后再今天买入(上一天卖出的最大利润减去今天的价格)之间的较大者。

  1. 第二次卖出 (j = 4):
dp[i][4]=max(dp[i−1][4],dp[i−1][3]+prices[i])

第二次卖出的最大利润是上一天第二次卖出的最大利润和上一天第二次买入后今天卖出(上一天第二次买入的最大利润加上今天的价格)之间的较大者。

通过这些递推公式,我们能够从第一天遍历到最后一天,同时更新每天的五个状态,最终得到完成两次交易的最大利润,即 dp[n-1][4],其中 n 是给定价格数组的长度。

1.1.3、 dp数组如何初始化

在第一天,我们可以选择买入股票,所以 dp[0][1] 和 dp[0][3] 都被初始化为 -prices[0],
这表示第一天买入股票的成本。其他状态的利润在第一天都是0(因为我们还没有进行任何卖出操作)

1.1.4、确定遍历顺序

从小到大遍历

1.2、代码

class Solution {public int maxProfit(int[] prices) {int len = prices.length;if (prices.length == 0)return 0;/** 定义 5 种状态:* 0: 没有操作, 1: 第一次买入, 2: 第一次卖出, 3: 第二次买入, 4: 第二次卖出*/int[][] dp = new int[len][5];dp[0][1] = -prices[0];dp[0][3] = -prices[0];for (int i = 1; i < len; i++) {dp[i][1] = Math.max(dp[i - 1][1], -prices[i]);dp[i][2] = Math.max(dp[i - 1][2], dp[i - 1][1] + prices[i]);dp[i][3] = Math.max(dp[i - 1][3], dp[i - 1][2] - prices[i]);dp[i][4] = Math.max(dp[i - 1][4], dp[i - 1][3] + prices[i]);}return dp[len - 1][4];}
}

1.2.1、如何理解这段代码

  1. 问题定义:给定一个数组 prices,其中 prices[i] 是第 i 天的股票价格。目标是找出最多进行两次交易(买入和卖出算一次交易)能获得的最大利润。注意,第二次买入必须在第一次卖出之后。

  2. 动态规划状态定义:定义一个二维数组 dp[len][5],其中 len 是给定价格数组的长度。dp[i][j] 表示第 i 天结束时处于五种状态之一的最大利润:

    • j = 0:没有进行任何操作。
    • j = 1:进行了第一次买入操作。
    • j = 2:进行了第一次卖出操作。
    • j = 3:进行了第二次买入操作。
    • j = 4:进行了第二次卖出操作。
  3. 初始化:在第一天,我们可以选择买入股票,所以 dp[0][1]dp[0][3] 都被初始化为 -prices[0],这表示第一天买入股票的成本。其他状态的利润在第一天都是0(因为我们还没有进行任何卖出操作),但代码中省略了这部分初始化,因为Java数组默认初始化为0。

  4. 状态转移方程:从第二天开始,我们根据前一天的状态更新当前天的状态:

    • dp[i][1]:第一次买入的最大利润,可能是前一天已经买入,或者今天才买入,取二者的较大值。
    • dp[i][2]:第一次卖出的最大利润,可能是前一天已经卖出,或者今天卖出(今天卖出的话,利润是前一天买入的利润加上今天的价格)。
    • dp[i][3]:第二次买入的最大利润,可能是前一天已经买入,或者今天买入(今天买入的话,利润是前一天第一次卖出的利润减去今天的价格)。
    • dp[i][4]:第二次卖出的最大利润,可能是前一天已经卖出,或者今天卖出(今天卖出的话,利润是前一天第二次买入的利润加上今天的价格)。
  5. 输出:最终的答案是 dp[len - 1][4],即最后一天结束时,完成两次买卖操作的最大利润。

通过动态规划表来追踪并更新每一天可能的最大利润,从而找到最多进行两次交易所能获得的最大利润。

二、股票买卖的最佳时机IV–至多K次买入和卖出

188.原题链接

2.1、动态规划五部曲

2.1.1、 确定dp数组(dp table)以及下标的含义

  • dp[len][k*2 + 1]:一个二维数组,len 是天数,k*2 + 1 是可能的交易状态数量。由于每次交易包括买入和卖出两个操作,因此共有 k*2 个操作状态,加上一个初始的没有操作状态,总共是 k*2 + 1 状态。
  • 对于 dp[i][j]
    • j 为0时,表示没有进行任何操作。
    • j 为奇数时,表示第 (j+1)/2 次交易持有股票(即买入状态)。
    • j 为偶数时,表示第 j/2 次交易不持有股票(即卖出状态)。

2.1.2、确定递推公式

对于第 i 天(i > 0),遍历所有交易状态 j,根据 j 的奇偶性,分别更新买入状态和卖出状态的最大利润:

  • 买入状态j 为奇数):dp[i][j] = Math.max(dp[i - 1][j], dp[i - 1][j - 1] - prices[i])
    • 如果在第 i 天买入股票,则最大利润是前一天同状态的利润和前一天卖出状态的利润减去今天的价格中的较大值。
  • 卖出状态j 为偶数):dp[i][j] = Math.max(dp[i - 1][j], dp[i - 1][j - 1] + prices[i])
    • 如果在第 i 天卖出股票,则最大利润是前一天同状态的利润和前一天买入状态的利润加上今天的价格中的较大值。

2.1.3、 dp数组如何初始化

对于第一天(i=0):

  • 没有进行任何交易的利润是0,即 dp[0][0] = 0
  • 对于所有的买入状态(j 为奇数),初始化为 -prices[0],因为如果在第一天买入股票,则损失了 prices[0] 的金额。

2.1.4、确定遍历顺序

从小到大遍历

2.1.5、输出结果

最终的结果是 dp[len - 1][k*2],即最后一天,进行完 k 次交易(全部卖出)后的最大利润。

通过这种方法,代码动态地计算在每一天结束时,根据不同交易次数和买卖状态下的最大利润,最终找到在给定交易次数限制下能够获得的最大利润。

2.2、代码

class Solution {public int maxProfit(int k, int[] prices) {if (prices.length == 0) return 0;// [天数][股票状态]// 股票状态: 奇数表示第 k 次交易持有/买入, 偶数表示第 k 次交易不持有/卖出, 0 表示没有操作int len = prices.length;int[][] dp = new int[len][k*2 + 1];for (int i = 1; i < k*2; i += 2) {dp[0][i] = -prices[0];}for (int i = 1; i < len; i++) {for (int j = 0; j < k*2 - 1; j += 2) {dp[i][j + 1] = Math.max(dp[i - 1][j + 1], dp[i - 1][j] - prices[i]);dp[i][j + 2] = Math.max(dp[i - 1][j + 2], dp[i - 1][j + 1] + prices[i]);}}return dp[len - 1][k*2];}
}

2.2.1、代码抄的,太难了

总结

1.感想

  • 2题hard题,都不会写,但是一看题解,好像也不是完全写不来的感觉。。

2.思维导图

本文思路引用自代码随想录,感谢代码随想录作者。

这篇关于代码随想录刷题day50|股票买卖的最佳时机III(至多2次买入和卖出)股票买卖的最佳时机IV(至多K次买入和卖出)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

随想录 Day 69 并查集 107. 寻找存在的路径

随想录 Day 69 并查集 107. 寻找存在的路径 理论基础 int n = 1005; // n根据题目中节点数量而定,一般比节点数量大一点就好vector<int> father = vector<int> (n, 0); // C++里的一种数组结构// 并查集初始化void init() {for (int i = 0; i < n; ++i) {father[i] = i;}

uniapp接入微信小程序原生代码配置方案(优化版)

uniapp项目需要把微信小程序原生语法的功能代码嵌套过来,无需把原生代码转换为uniapp,可以配置拷贝的方式集成过来 1、拷贝代码包到src目录 2、vue.config.js中配置原生代码包直接拷贝到编译目录中 3、pages.json中配置分包目录,原生入口组件的路径 4、manifest.json中配置分包,使用原生组件 5、需要把原生代码包里的页面修改成组件的方

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

17.用300行代码手写初体验Spring V1.0版本

1.1.课程目标 1、了解看源码最有效的方式,先猜测后验证,不要一开始就去调试代码。 2、浓缩就是精华,用 300行最简洁的代码 提炼Spring的基本设计思想。 3、掌握Spring框架的基本脉络。 1.2.内容定位 1、 具有1年以上的SpringMVC使用经验。 2、 希望深入了解Spring源码的人群,对 Spring有一个整体的宏观感受。 3、 全程手写实现SpringM

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std

记录AS混淆代码模板

开启混淆得先在build.gradle文件中把 minifyEnabled false改成true,以及shrinkResources true//去除无用的resource文件 这些是写在proguard-rules.pro文件内的 指定代码的压缩级别 -optimizationpasses 5 包明不混合大小写 -dontusemixedcaseclassnames 不去忽略非公共

麻了!一觉醒来,代码全挂了。。

作为⼀名程序员,相信大家平时都有代码托管的需求。 相信有不少同学或者团队都习惯把自己的代码托管到GitHub平台上。 但是GitHub大家知道,经常在访问速度这方面并不是很快,有时候因为网络问题甚至根本连网站都打不开了,所以导致使用体验并不友好。 经常一觉醒来,居然发现我竟然看不到我自己上传的代码了。。 那在国内,除了GitHub,另外还有一个比较常用的Gitee平台也可以用于

LeetCode--220 存在重复元素 III

题目 给定一个整数数组,判断数组中是否有两个不同的索引 i 和 j,使得 nums [i] 和 nums [j] 的差的绝对值最大为 t,并且 i 和 j 之间的差的绝对值最大为 ķ。 示例 示例 1:输入: nums = [1,2,3,1], k = 3, t = 0输出: true示例 2:输入: nums = [1,0,1,1], k = 1, t = 2输出: true示例

众所周知,配置即代码≠基础设置即代码

​前段时间翻到几条留言,问: “配置即代码和基础设施即代码一样吗?” “配置即代码是什么?怎么都是基础设施即代码?” 我们都是知道,DevOp的快速发展,让服务器管理与配置的时间大大减少,配置即代码和基础设施即代码作为DevOps的重要实践,在其中起到了关键性作用。 不少人将二者看作是一件事,配置即大代码是关于管理特定的应用程序配置设置本身,而基础设施即代码更关注的是部署支持应用程序环境所需的

53、Flink Interval Join 代码示例

1、概述 interval Join 默认会根据 keyBy 的条件进行 Join 此时为 Inner Join; interval Join 算子的水位线会取两条流中水位线的最小值; interval Join 迟到数据的判定是以 interval Join 算子的水位线为基准; interval Join 可以分别输出两条流中迟到的数据-[sideOutputLeftLateData,