代码随想录-刷题第四十九天

news2025/1/13 6:04:16

121. 买卖股票的最佳时机

题目链接:121. 买卖股票的最佳时机

思路:动态规划五步曲

  1. dp[i][0] 表示第i天持有股票所得最多现金,dp[i][1] 表示第i天不持有股票所得最多现金。

    一开始现金是0,那么加入第i天买入股票,现金就是 -prices[i], 这是一个负数。

    注意这里说的是“持有”,“持有”不代表就是当天“买入”!也有可能是昨天就买入了,今天保持持有的状态

  2. 递推公式:

    如果第i天持有股票即dp[i][0], 那么可以由两个状态推出来

    第i-1天就持有股票,那么就保持现状,所得现金就是昨天持有股票的所得现金 即:dp[i - 1][0]

    第i天买入股票,所得现金就是买入今天的股票后所得现金即:-prices[i]

    那么dp[i][0]应该选所得现金最大的,

    dp[i][0] = max(dp[i - 1][0], -prices[i])

    如果第i天不持有股票即dp[i][1], 也可以由两个状态推出来

    第i-1天就不持有股票,那么就保持现状,所得现金就是昨天不持有股票的所得现金 即:dp[i - 1][1]

    第i天卖出股票,所得现金就是按照今天股票价格卖出后所得现金 即:dp[i - 1][0] + prices[i]

    同样dp[i][1]取最大的,

    dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] + prices[i])

  3. 初始化:

    由递推公式可以看出其基础都是要从dp[0][0]和dp[0][1]推导出来。

    那么dp[0][0]表示第0天持有股票,此时的持有股票就一定是买入股票了,因为不可能由前一天推出来,所以dp[0][0] = -prices[0]

    dp[0][1]表示第0天不持有股票,不持有股票即现金就是0,所以dp[0][1] = 0

  4. 遍历顺序:由递推公式可以看出,需要从前向后遍历。

  5. 举例推导dp数组

    以输入:[7,1,5,3,6,4]为例,dp数组状态如下:

    121.买卖股票的最佳时机

    dp[5][1]就是最终结果。为什么不是dp[5][0]呢?

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

class Solution {
    public int maxProfit(int[] prices) {
        int len = prices.length;
        // dp[i][0]代表第i天持有股票的最大收益
        // dp[i][1]代表第i天不持有股票的最大收益
        int[][] dp = new int[len][2];
        
        // 初始化
        dp[0][0] = -prices[0];
        dp[0][1] = 0;
        for (int i = 1; i < len; i++) {
            dp[i][0] = Math.max(dp[i - 1][0], -prices[i]);
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] + prices[i]);
        }
        return dp[len - 1][1];
    }
}

122. 买卖股票的最佳时机 II

题目链接:122. 买卖股票的最佳时机 II

思路:本题和121. 买卖股票的最佳时机的唯一区别是本题股票可以买卖多次了(注意任何时候 最多 只能持有 一股 股票,所以再次购买前要出售掉之前的股票)

在动规五步曲中,这个区别主要是体现在递推公式上,其他都和上一题相同

dp数组的含义:

  • dp[i][0] 表示第i天持有股票所得现金
  • dp[i][1] 表示第i天不持有股票所得最多现金

如果第i天持有股票即dp[i][0], 那么可以由两个状态推出来

  • 第i-1天就持有股票,那么就保持现状,所得现金就是昨天持有股票的所得现金 即:dp[i - 1][0]
  • 第i天买入股票,所得现金就是昨天不持有股票的所得现金减去今天的股票价格 即:dp[i - 1][1] - prices[i]

在121. 买卖股票的最佳时机中,因为股票全程只能买卖一次,所以如果买入股票,那么第i天持有股票即dp[i][0]一定就是 -prices[i]。

而本题,因为一只股票可以买卖多次,所以当第i天买入股票的时候,所持有的现金可能有之前买卖过的利润。

再来看看如果第i天不持有股票即dp[i][1]的情况, 依然可以由两个状态推出来

  • 第i-1天就不持有股票,那么就保持现状,所得现金就是昨天不持有股票的所得现金 即:dp[i - 1][1]
  • 第i天卖出股票,所得现金就是按照今天股票价格卖出后所得现金即:dp[i - 1][0] + prices[i]

注意这里与上一题是一样的逻辑,卖出股票收获利润(可能是负值)天经地义!

class Solution {
    public int maxProfit(int[] prices) {
        int len = prices.length;
        // dp[i][0]代表第i天持有股票的最大收益
        // dp[i][1]代表第i天不持有股票的最大收益
        int[][] dp = new int[len][2];
        
        // 初始化
        dp[0][0] = -prices[0];
        dp[0][1] = 0;
        for (int i = 1; i < len; i++) {
            // 注意这里是和121. 买卖股票的最佳时机唯一不同的地方。
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][1] - prices[i]);
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] + prices[i]);
        }
        return dp[len - 1][1];
    }
}

123. 买卖股票的最佳时机 III

题目链接:123. 买卖股票的最佳时机 III

思路:这道题目相对前两题难了不少。关键在于至多买卖两次,这意味着可以买卖一次,可以买卖两次,也可以不买卖。

动态规划五步曲:

  1. 确定dp数组以及下标的含义

    一天有五个状态

    1. 没有操作 (其实也可以不设置这个状态)

    2. 第一次持有股票

    3. 第一次不持有股票

    4. 第二次持有股票

    5. 第二次不持有股票

    dp[i][j]中 i表示第i天,j为 [0 - 4] 五个状态,dp[i][j]表示第i天状态j所剩最大现金。

    需要注意:dp[i][1]表示的是第i天,买入股票的状态,并不是说一定要第i天买入股票,这是容易陷入的误区

    例如 dp[i][1],并不是说第i天一定买入股票,有可能第i-1天就买入了,那么 dp[i][1]延续买入股票的这个状态。

  2. 确定递推公式

    达到dp[i][1]状态,有两个具体操作:

    操作一:第i天买入股票了,那么dp[i][1] = dp[i - 1][0] - prices[i]

    操作二:第i天没有操作,沿用前一天买入的状态,即:dp[i][1] = dp[i - 1][1]

    那么dp[i][1]究竟选 dp[i - 1][0] - prices[i],还是dp[i - 1][1]呢?

    一定是选最大的,dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] - prices[i])

    同理dp[i][2]也有两个操作:

    操作一:第i天卖出股票了,那么dp[i][2] = dp[i - 1][1] + prices[i]

    操作二:第i天没有操作,沿用前一天卖出的状态,即:dp[i][2] = dp[i - 1][2]

    所以dp[i][2] = max(dp[i - 1][2], dp[i - 1][1] + prices[i])

    同理可推出剩下状态部分:

    dp[i][3] = max(dp[i - 1][3], dp[i - 1][2] - prices[i])

    dp[i][4] = max(dp[i - 1][4], dp[i - 1][3] + prices[i])

  3. dp数组初始化

    第0天没有操作,这个最容易想到,就是0,即:dp[0][0] = 0

    第0天做第一次买入的操作,dp[0][1] = -prices[0]

    第0天做第一次卖出的操作,这个初始值应该是多少呢?

    此时还没有买入,怎么就卖出呢?

    其实可以理解当天买入,当天卖出,所以dp[0][2] = 0

    第0天第二次买入操作,初始值应该是多少呢?

    第一次还没买入呢,怎么初始化第二次买入呢?

    第二次买入依赖于第一次卖出的状态,其实相当于第0天第一次买入了,第一次卖出了,然后再买入一次(第二次买入),那么现在手头上没有现金,只要买入,现金就相应减少。所以第二次买入操作,初始化为:dp[0][3] = -prices[0]

    同理第二次卖出初始化dp[0][4] = 0

  4. 确定遍历顺序

    由递推公式可以看出,一定是从前向后遍历,因为dp[i],依靠dp[i - 1]的数值。

  5. 举例推导dp数组

    以输入[1,2,3,4,5]为例

    123.买卖股票的最佳时机III

    大家可以看到红色框为最后两次卖出的状态。

    现在最大的时候一定是卖出的状态,而两次卖出的状态现金最大的一定是最后一次卖出。

    也可以这么理解:如果第一次卖出已经是最大值了,那么可以在当天立刻买入再立刻卖出。所以dp[4][4]已经包含了dp[4][2]的情况。也就是说第二次卖出手里所剩的钱一定是最多的。所以最终最大利润是dp[4][4]

class Solution {
    public int maxProfit(int[] prices) {
        int len = prices.length;
        /*
         * 定义 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][0] = dp[i - 1][0];
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] - 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];
    }
}

188. 买卖股票的最佳时机 IV

题目链接:188. 买卖股票的最佳时机 IV

思路:本题是上一题的进阶版,这里要求至多有k次交易。

动态规划五步曲:

  1. 确定dp数组以及下标的含义

    使用二维数组 dp[i][j] :第i天的状态为j,所剩下的最大现金是dp[i][j]

    j的状态表示为:

    • 0 表示不操作
    • 1 第一次买入
    • 2 第一次卖出
    • 3 第二次买入
    • 4 第二次卖出

    可以发现规律 ,除了0以外,奇数就是买入,偶数就是卖出。

    题目要求是至多有K笔交易,那么j的范围定义为 2 * k + 1 就可以了。

  2. 确定递推公式

    需要强调一下:dp[i][1]表示的是第i天,买入股票的状态,并不是说一定要第i天买入股票,这是容易陷入的误区

    达到dp[i][1]状态,有两个具体操作:

    操作一:第i天买入股票了,那么dp[i][1] = dp[i - 1][0] - prices[i]

    操作二:第i天没有操作,沿用前一天买入的状态,即:dp[i][1] = dp[i - 1][1]

    选最大的,所以 dp[i][1] = max(dp[i - 1][1], dp[i - 1][0] - prices[i])

    同理dp[i][2]也有两个操作:

    操作一:第i天卖出股票了,那么dp[i][2] = dp[i - 1][1] + prices[i]

    操作二:第i天没有操作,沿用前一天卖出的状态,即:dp[i][2] = dp[i - 1][2]

    所以dp[i][2] = max(dp[i - 1][2], dp[i - 1][1] + prices[i])

    同理可以类比剩下的状态,代码如下:

    for (int i = 1; i < prices.size(); i++) {
        for (int j = 0; j < 2 * k - 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]);
        }
    }
    

    本题与上一题最大的区别就是这里要类比j为奇数是买,偶数是卖的状态

  3. dp数组初始化

    第0天没有操作,这个最容易想到,就是0,即:dp[0][0] = 0

    第0天做第一次买入的操作,dp[0][1] = -prices[0]

    第0天做第一次卖出的操作,这个初始值应该是多少呢?

    此时还没有买入,怎么就卖出呢?

    其实可以理解当天买入,当天卖出,所以dp[0][2] = 0

    第0天第二次买入操作,初始值应该是多少呢?

    第一次还没买入呢,怎么初始化第二次买入呢?

    第二次买入依赖于第一次卖出的状态,其实相当于第0天第一次买入了,第一次卖出了,然后再买入一次(第二次买入),那么现在手头上没有现金,只要买入,现金就相应减少。所以第二次买入操作,初始化为:dp[0][3] = -prices[0]

    同理第二次卖出初始化dp[0][4] = 0

    可以推出dp[0][j]当j为奇数的时候都初始化为 -prices[0]

    代码如下:

    for (int j = 1; j < 2 * k; j += 2) {
        dp[0][j] = -prices[0];
    }
    

    在初始化的地方同样要类比j为奇数是买,偶数是卖的状态。

  4. 确定遍历顺序

    由递推公式可以看出,一定是从前向后遍历,因为dp[i],依靠dp[i - 1]的数值。

  5. 举例推导dp数组

    以输入[1,2,3,4,5],k=2为例。

    188.买卖股票的最佳时机IV

    最后一次卖出,一定是利润最大的,dp[len - 1][2 * k]即红色部分就是最终结果。

class Solution {
    public int maxProfit(int k, int[] prices) {
        int len = prices.length;
        if (len == 0) return 0;
        
        // 至多有K笔交易,那么j的范围定义为 2 * k + 1 
        int[][] dp = new int[len][2 * k + 1];
        
        // 初始化
        for (int j = 1; j < 2 * k; j += 2) {
            // dp[0][j]当j为奇数的时候都初始化为 -prices[0]
            dp[0][j] = -prices[0];
        }

        for (int i = 1; i < len; i++) {
            for (int j = 0; j < 2 * k - 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][2 * k];
    }
}

也可以使用三维dp数组来解题

class Solution {
    public int maxProfit(int k, int[] prices) {
        int len = prices.length;
        if (len == 0) return 0;

        // [天数][交易次数][是否持有股票]
        int[][][] dp = new int[len][k + 1][2];
        
        // dp数组初始化
        for (int i = 0; i <= k; i++) {
            dp[0][i][1] = -prices[0];
        }

        for (int i = 1; i < len; i++) {
            for (int j = 1; j <= k; j++) {
                // dp方程, 0表示不持有/卖出, 1表示持有/买入
                dp[i][j][0] = Math.max(dp[i - 1][j][0], dp[i - 1][j][1] + prices[i]);
                dp[i][j][1] = Math.max(dp[i - 1][j][1], dp[i - 1][j - 1][0] - prices[i]);
            }
        }
        return dp[len - 1][k][0];
    }
}

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

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

相关文章

深入了解 RDD

深入了解 RDD 案例 明确需求&#xff1a; 在访问日志中&#xff0c;统计独立IP数量 TOP10 查看数据结构&#xff1a; IP&#xff0c;时间戳&#xff0c;Http&#xff0c;Method&#xff0c;Url…… 明确编码步骤 取出IP&#xff0c;生成一个只有IP的数据集简单清洗统计IP出现…

Tomcat Notes: Deployment File

This is a personal study notes of Apache Tomcat. Below are main reference material. - YouTube Apache Tomcat Full Tutorial&#xff0c;owed by Alpha Brains Courses. https://www.youtube.com/watch?vrElJIPRw5iM&t801s 1、Tomcat deployment1.1、Two modes of …

数据结构与算法教程,数据结构C语言版教程!(第二部分、线性表详解:数据结构线性表10分钟入门)九

第二部分、线性表详解&#xff1a;数据结构线性表10分钟入门 线性表&#xff0c;数据结构中最简单的一种存储结构&#xff0c;专门用于存储逻辑关系为"一对一"的数据。 线性表&#xff0c;基于数据在实际物理空间中的存储状态&#xff0c;又可细分为顺序表&#xff…

Java 基础知识点1 (含面试题)

本次Java 知识点主要是关于SE的相关基础&#xff0c;同时也包含了数据结构中的一些API&#xff0c;例如Set,List,Map等&#xff0c;最后也附上了相关重要的面试题&#xff0c;可供大家学习与参考&#xff01; 目录 重要知识点数据结构API面试题 重要知识点 Java 是一门面向对象…

linux下超级程序!在linux界面实现类图像化界面的操作体验!

linux下超级程序&#xff01;在linux界面实现类图像化界面的操作体验&#xff01; 本期带来一个超级程序&#xff01;在linux界面实现类图像化界面的操作体验。具体功能代码如下: 1500行完整代码想要完成部署&#xff0c;只需在本地创建一个LinuxGJ.sh的文件&#xff0c;然后…

传感数据分析——高通滤波与低通滤波

传感数据分析——高通滤波与低通滤波 文章目录 传感数据分析——高通滤波与低通滤波前言一、运行环境二、Python实现总结 前言 对于传感信号而言&#xff0c;我们可以提取其中的高频信息和低频信息&#xff0c;低频信息往往是信号的趋势&#xff0c;高频信息往往是一些突变或异…

构建自己的私人GPT

创作不易&#xff0c;请大家多鼓励支持。 在现实生活中&#xff0c;很多人的资料是不愿意公布在互联网上的&#xff0c;但是我们又要使用人工智能的能力帮我们处理文件、做决策、执行命令那怎么办呢&#xff1f;于是我们构建自己或公司的私人GPT变得非常重要。 一、本地部署…

win10下vscode+cmake编译C代码操作详解

0 工具准备 1.Visual Studio Code 1.85.1 2.cmake 3.24.01 前言 当我们只有一个.c文件时直接使用vscodeCode Runner插件即可完成编译&#xff0c;如果我们的工程很复杂包含多个.c文件时建议使用cmake来生成对应的make&#xff0c;指导编译器完成编译&#xff0c;否则会提示各…

数字图像处理(图像灰度变换、图像直方图及均衡、图像中值滤波、图像空域锐化增强、图像频域滤波)

数字图像处理&#xff08;图像灰度变换、图像直方图及均衡、图像中值滤波、图像空域锐化增强、图像频域滤波&#xff09; 目录 1 图像灰度变换 1.1 灰度线性变换 1.2 图像二值化 1.3 负象变换 1.4 灰度非线性变换 1.5 程序设计流程图 2 图像直方图及均衡 2.1 直方图 2…

无心剑七绝《高斯黎曼》

七绝高斯黎曼 高耸云端四海惊 斯人伟绩震豪英 黎霞璀璨通灵处 曼妙方程万世名 2024年1月6日 平水韵八庚平韵 《七绝高斯黎曼》是无心剑所作的一首以数学家为主题的七言绝句。全诗巧妙地将两位杰出的数学家——高斯&#xff08;Carl Friedrich Gauss&#xff09;与黎曼&#…

大模型机器人原理解析:如何从RT/RT2、Berkeley Gello到发展到斯坦福Mobile ALOHA、Google家务机器人

前言 23年7月&#xff0c;我在朋友圈评估Google的RT2说道&#xff1a; “大模型正在革新一切领域啊&#xff0c;超帅&#xff0c;通过大模型不仅能理解“人话”&#xff0c;还能对“人话”进行推理&#xff0c;并转变为机器人能理解的指令&#xff0c;从而分阶段完成任务。回…

express框架

目录 一、express介绍二、express 使用2.1 express下载2.2 express初体验 三、express 路由3.1 路由的使用3.2 获取参数3.3 获取路由参数 四、express响应设置五、express中间件5.1 什么是中间件5.2 中间件的作用5.3 中间件的类型5.3.1 定义全局中间件5.3.2 多个全局中间件5.3.…

协程池与新脚本语言

今天的主人公名为——Melang。 这是一款使用C语言开发的“新”的脚本语言&#xff0c;然而其已经默默问世了6年之久。 下面笔者就带你走进Melang world。 What is Melang Melang是一款协程并发脚本语言。它是一款解释型&#xff0c;而非编译型语言。 在Melang中&#xff…

html中的form表单以及相关控件input、文本域、下拉select等等的详细解释 ,点赞加关注持续更新~

文章目录 表单创建表单forminput 标签input标签的value属性设置input标签格式单选框多选框上传文件下拉菜单文本域设置文本域格式label 标签按钮 表单 作用&#xff1a;收集用户信息。 使用场景&#xff1a; 登录页面注册页面搜索区域 创建表单form <form action".…

实战环境搭建-linux下安装jdk1.8

查看安装jdk版本信息,主要是怕之前有遗漏的,或者安装失败的java rpm -qa | grep java 显示如下信息: 卸载: rpm -e --nodeps java-1.7.0-openjdk-1.7.0.261-2.6.22.2.el7_8.x86_64 rpm -e --nodeps java-1.8.0-openjdk-1.8.0.262.b10-1.el7.x86_64 还有一些其他的命令…

MATLAB插值函数

一、MATLAB插值函数概览 1&#xff09;本节重点介绍的插值函数 MATLAB插值函数适用情况基础句式interp1 函数interp1 主要用于一维数据的插值interp1(x, y, x_interp, ‘linear’); 其中 x 和 y 是已知数据点&#xff0c;x_interp 是要插值的目标点。interp2 函数interp2 用于…

【STM32】PWR电源控制

1 PWR简介 PWR&#xff08;Power Control&#xff09;电源控制 PWR负责管理STM32内部的电源供电部分&#xff0c;可以实现可编程电压监测器和低功耗模式的功能 可编程电压监测器&#xff08;PVD&#xff09;可以监控VDD电源电压&#xff0c;当VDD下降到PVD阀值以下或上升到P…

回首2023,期待2024!

2023&#xff0c;在改变中到来 2023年1月1日&#xff0c;我从成都冷清的学校回到了哈尔滨的老家&#xff0c;开始了保研之前的最后一个寒假 当时的目标是将之前的科研理论转化为实际&#xff0c;生产出一篇sci&#xff0c;助力保研加分 星移斗转&#xff0c;事与愿违&#x…

从零实现CLIP模型

1. 引言 CLIP代表语言图像对比预训练模型&#xff0c;是OpenAI于2021年开发的一个深度学习模型。CLIP模型中图像和文本嵌入共享相同的潜在特征空间&#xff0c;从而能够在两种模式之间直接进行对比学习。这是通过训练模型使相关的图像和文本更紧密地结合在一起&#xff0c;同时…

MySQL高级DBA的理论与实践,MySQL数据库管理员从入门到精通

一、教程描述 数据库管理员&#xff08;Database Administrator&#xff09;&#xff0c;简称DBA&#xff0c;想要成为高级的MySQL DBA&#xff0c;就要耐得住寂寞&#xff0c;持续不断地学习&#xff0c;除了数据库专业知识外&#xff0c;还需要了解主机、系统、网络、存储、…