首页 > 其他分享 >【动态规划】No 309. 最佳买卖股票时机含冷冻期

【动态规划】No 309. 最佳买卖股票时机含冷冻期

时间:2023-05-03 14:00:10浏览次数:49  
标签:309 No max dp 状态 prices 股票 冷冻 Math

【动态规划】309. 最佳买卖股票时机含冷冻期

给定一个整数数组prices,其中第 prices[i] 表示第 i 天的股票价格 。

设计一个算法计算出最大利润。在满足以下约束条件下,你可以尽可能地完成更多的交易(多次买卖一支股票):

卖出股票后,你无法在第二天买入股票 (即冷冻期为 1 天)。
注意:你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。

示例 1:

输入: prices = [1,2,3,0,2]
输出: 3 
解释: 对应的交易状态为: [买入, 卖出, 冷冻期, 买入, 卖出]

示例 2:

输入: prices = [1]
输出: 0

提示:

  • 1 <= prices.length <= 5000
  • 0 <= prices[i] <= 1000
两状态解法

​ 思路:加了冰冻期后,对于当前没有股票的状态是没有影响的,仍是由昨天持有股票今天卖出昨天不持有股票两种状态转化而来:

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

​ 而对于当前持有股票却不能直接使用昨天不持有股票的状态然后今天购买了,因为有可能昨天卖出了股票导致今天不能购买,所以这部分代码为:

dp[i][1] = Math.max(dp[i-2][0] - prices[i], dp[i-1][1]);

​ 看到这可能会有疑问,这个只是包含前天没有股票的状态,如果昨天没有交易而且也没有股票岂不是没有统计?我们只是不需要昨天交易导致没有股票这一个状态而已。这是由于昨天没有交易而且也没有股票这一状态一定是由前天没有股票转移过来的,因为昨天有交易的话,前天是一定有股票的。

class Solution {
    public int maxProfit(int[] prices) {
        var len = prices.length;
        var dp = new int[len][2];
        dp[0][0] = 0;
        dp[0][1] = -prices[0];

        for(var i = 1; i < len; i++) {
            dp[i][0] = Math.max(dp[i-1][1] + prices[i], dp[i-1][0]);
            if(i > 1) {
                dp[i][1] = Math.max(dp[i-2][0] - prices[i], dp[i-1][1]);
            } else {
                dp[i][1] = Math.max(dp[i-1][0] - prices[i], dp[i-1][1]);
            }
        }

        return dp[len-1][0];
    }
}
四状态解法

​ 整个过程可以分为四个状态:

  • 状态一:持有股票的状态,今天买入股票 或 之前就买入了股票但是一直没有卖出
  • 状态二:不持有股票的状态,且保持卖出股票的状态,即两天前就卖出了股票一直没有操作
  • 状态三:不持有股票的状态,且是今天卖出的股票
  • 状态四:不持有股票的状态,冰冻期,即一天前卖出了股票

​ 则递推可以写为:

			// 持有股票的状态,由状态1、3转化而来,2状态今天是冰封期
            dp[i][0] = Math.max(dp[i-1][0], Math.max(dp[i-1][1], dp[i-1][3]) - prices[i]);
            // 不持有,且是两天前卖的,由更多天前卖的和3状态转化而来,2状态今天也是冰封期
            dp[i][1] = Math.max(dp[i-1][1], dp[i-1][3]);
            // 不持有,今天卖的
            dp[i][2] = dp[i-1][0] + prices[i];
            // 不持有,且是一天前卖的,今天是冰封期
            dp[i][3] = dp[i-1][2];

​ 代码为:

class Solution {
    public int maxProfit(int[] prices) {
        var len = prices.length;
        var dp = new int[len][4];
        dp[0][0] = -prices[0];
        for(var i = 1; i < len; i++) {
            // 持有股票的状态
            dp[i][0] = Math.max(dp[i-1][0], Math.max(dp[i-1][1], dp[i-1][3]) - prices[i]);
            // 不持有,且是二天前卖的
            dp[i][1] = Math.max(dp[i-1][1], dp[i-1][3]);
            // 不持有,今天卖的
            dp[i][2] = dp[i-1][0] + prices[i];
            // 不持有,且是一天前卖的,今天是冰封期
            dp[i][3] = dp[i-1][2];
        }
        return Math.max(Math.max(dp[len-1][1], dp[len-1][2]), dp[len-1][3]);
    }
}
一维数组优化

​ 思路是不变的,只是注意下面的定义和赋值的顺序,否则容易被覆盖修改

class Solution {
    public int maxProfit(int[] prices) {
        var len = prices.length;
        var dp = new int[4];
        dp[0] = -prices[0];

        for(var i = 1; i < len; i++) {
            // 持有状态
            dp[0] = Math.max(dp[0], Math.max(dp[1], dp[2]) - prices[i]);
            // 不持有状态,且是两天前卖的
            dp[1] = Math.max(dp[1], dp[2]);
            // 不持有状态,且是一天前卖的
            dp[2] = dp[3];
            // 不持有状态,且是当天卖的
            dp[3] = dp[0] + prices[i];
        }

        // System.out.println(Arrays.toString(dp));

        return Math.max(Math.max(dp[1], dp[2]), dp[3]);
    }
}

标签:309,No,max,dp,状态,prices,股票,冷冻,Math
From: https://www.cnblogs.com/tod4/p/17368994.html

相关文章

  • idea创建SpringBoot项目报错For artifact {mysql:mysql-connector-java:null:jar}: Th
    Forartifact{mysql:mysql-connector-java:null:jar}:Theversioncannotbeempty.报错如图:pom.xml文件如图:添加版本号:就好了......
  • P4211 [LNOI2014]LCA
    \(\color{purple}\text{P4211[LNOI2014]LCA}\)解题方法可以发现一个结论:两个点到根节点的重合路径的长度即为他们\(LCA\)的深度。所以我们把\([l,r]\)之间的点到根节点路径上各加一,再查询\(z\)到根节点的路径的值之和即为\(\sum_{i=l}^{r}\text{dep}[\text{LCA}(i,z)]\)......
  • 专访Evernote CEO Phil Libin:Evernote想留住你的记忆而不是你的Money
    在伦敦举行的 LeWeb大会以后,Evernote CEO PhilLibin 提到,Evernote现在的用户数量已从五月份的2500万上升到六月份的3400万,付费用户虽然比率仍旧很小,但也在快速增长,从五月份的100万增长到六月份的140万。就Evernote今后的发展计划,著名科技博客TechCrunch编辑INGRIDLUNDEN对Ev......
  • 「NOIP2008」笨小猴
    笨小猴的词汇量很小,所以每次做英语选择题的时候都很头疼。但是他找到了一种方法,经试验证明,用这种方法去选择选项的时候选对的几率非常大!这种方法的具体描述如下:假设maxn是单词中出现次数最多的字母的出现次数,minn是单词中出现次数最少的字母的出现次数,如果maxn-minn是一个质......
  • go krotos proto编译引用外部包 was not found or had errors
    前言kratosprotos生成pb.go文件时,会出现引用其他proto文件报错wasnotfoundorhaderrors,因找不到此文件而无法编译。解决首先我们先了解下protoc中import的两条规则:import不允许使用相对路径;import导入路径应该从根开始的绝对路径这个根开始的绝对路径指......
  • 对外提供的api保证接口的幂等 (先select 再 update innodb是行级锁, mysam是表级的
    额外的状态字段,这个状态值一般只会单流程变更,不管通过什么消息传递,目前申万宏源的每一个业务大部分都走流程,走的过程就有唯一的业务字段配合工作流workflow服务来进行业务流转个人观点解决幂等只有两种方式第一种依赖上游带过来的唯一标志,然后我们给这个唯一标志加锁保证请......
  • 【pytorch】为什么 ToTensor 后紧接 Normalize 操作?
    学习pytorch的transforms一节中产生疑问:ToTensor操作中图像数据满足[0,255]条件会进行线性归一化,映射到[0,1]。在ToTensor操作后一般紧接着Nomalize操作,又进行了一次标准差归一化。既然已经归一化了一次,为什么还要再来一次?以下是我在网络上找到的一些答案:数据如果......
  • Notepad++替换(正则)
    Ctrl+H打开替换:行首空格和空行去除:"^\s+"->""行首插入ABCD:"^"->"ABCD"行尾空格和空行去除:"\s+$"->""行尾插入ABCD:"$"->"ABCD"按头AB,尾CD去除内容(单行非贪婪):"AB.+?CD"或"AB.*?CD......
  • CF1817C Similar Polynomials 题解
    可能更好的阅读体验题目传送门Div.1C拉格朗日差值,赛时开香槟。题目大意给定\(d\)次两个多项式\(A(x),B(x)\)。求\(s\),使得\(B(x)\equivA(x+s)\pmod{10^9+7}\),保证\(s\)存在。给出多项式的形式为给出\(x=0,1,\cdots,d\)模\(10^9+7\)的值。\(d\le2.5\times......
  • SpringBoot Maven打jar包提示no main manifest attribute springboot
    SpringBoot项目打jar包运行jar包提示:nomainmanifestattributespringbootpom依赖:<build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-m......