动规训练4

目录

一、买股票的最佳实际含冷冻期

1、题目解析

2、算法原理

a状态表示方程

b状态转移方程

c初始化

d填表顺序

e返回值

3、代码

4、感想

二、买股票的最佳时机函手续费

1、题目解析

2、算法原理

a状态表示方程

b状态转移方程

c初始化

d填表顺序

e返回值

3、代码

4、写题感悟

三、买卖股票最佳时机4

1、题目解析

2、算法原理

a状态表示方程

b状态转移方程

c初始化

d填表顺序

e返回值

3、代码


这篇文章每道题我都会以题目解析、算法原理、以及代码编写这三个模块进行讲解。

在算法原理这里,我们使用动态规划的解题模式进行讲解:

  1. 找出状态表示方程
  2. 找出状态转移方程
  3. 如何进行初始化
  4. dp表的填写顺序
  5. 该返回什么值

推荐从专题中的动态规划1开始进行训练,由浅入深

一、买股票的最佳实际含冷冻期

1、题目解析

这道题就是一个先买——>卖——>冷冻期——>买这三个过程然后期间可以选择何时卖也可以选择买,但是主体上就是按照这个流程来的。

而且通过示例二可以得出,只有一天交易的时候不要买入,不然会亏本

2、算法原理

a状态表示方程

经验+题目要求

dp[i]粗浅的表示当天结束后获得的最大利润

但是dp[i]会涵盖三种状态

  • 买入状态——手上已经买入了股票
  • 冷冻期状态——将股票卖出后进入的状态
  • 可交易状态——经过冷冻期可以选择对当天股票是否买入的状态

这三种情况我们需要分别讨论,就需要三个表来表示。

因此状态表示方程应该是:

dp[i][0]表示第i天结束后,处于已买入状态的最大利润。

dp[i][1]表示第i天结束后,处于可交易状态的最大利润。

dp[i][2]表示第i天结束后,处于冷冻期状态的最大利润。

b状态转移方程

根据最近的一步划分问题

这道题目复杂的地方在于,最近的一步也有三种情况,我们要分别讨论前一天的三种情况怎么转换成今天的三种情况之一。

以转换成买入这种状态我们就需要讨论三种情况,

然后另外拉警钟情况都需要分别讨论,但其实有些情况是不可能达成的,以上面为例:

  • 买入——>买入,啥事不做就可以达成了
  • 冷冻——>买入,不可能达成,前一天结束时冷冻期意味着今天就是冷冻期。
  • 可交易——>买入,前一天结束是可交易,那我今天买入股票,今天结束就是买入状态了。

所以最后只有这两种情况

讨论完:

所以状态转移方程为:

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

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

dp[i][2]=dp[i-1][0]+p[i]

c初始化

为了避免边界问题:

将dp[0][0],dp[0][1],dp[0][2]这三个值初始化,分别是:-p[i],0,0

d填表顺序

从左到右,每次每次填入上个表。

e返回值

返回最后一天三种情况最大值即可

3、代码

class Solution {
public:int maxProfit(vector<int>& prices) {//建表//初始化//填表//返回值int n=prices.size();int dp[n][3];dp[0][0]=0-prices[0];dp[0][1]=0;dp[0][2]=0;for(int i=1;i<n;i++){dp[i][0]=max(dp[i-1][0],dp[i-1][1]-prices[i]);dp[i][1]=max(dp[i-1][1],dp[i-1][2]);dp[i][2]=dp[i-1][0]+prices[i];}return max(max(dp[n-1][0],dp[n-1][1]),dp[n-1][2]);}
};

4、感想

写到这道题的时候我感觉动态规划的解题方法是将所有情况穷举完了,然后选取最合适的值,尽管我们知道有些地方可以优化,但是这种穷举的方法反而可以帮我们解决处理这些优化地方的思考。

就像是返回值,我们直觉上会觉得返回值那天肯定不需要购入股票了,我们压根就不需要考虑这情况。但是站在代码的层面上考虑的话,就是多加了一个比较的是,这样还不需要我们去考虑一些特殊情况。

在写一道题目时侯我们可以根据示例,将一道题准确的得出最优解,但是要我们总结其中逻辑时却无从下手,因为情况太多了,反正你是一步接着一步,我把每一步的情况考虑清楚的话是可以选出我们需要的结果的,这其实也是一种剪枝思想的实现。

二、买股票的最佳时机函手续费

1、题目解析

相较于第一道题,这里有三种状态——买入后、卖出后、可买入。不过在一个完整的交易过程后需要扣除一个手续费。

2、算法原理

a状态表示方程

经验+题目要求

dp[i]粗浅的表示第i天后获得的最大利润

不过第i天有可能有三种情况,买入后、卖出后、可买入

经过分析:

我们发现买入到达可买是没有任何数值变换的,所以其实这一种状态其实是没有 必要的。

简化成:

所以我们需要运营两个表:

dp[i][0]表示第i天结束后为未持有股票的情况下拥有的最高利润

dp[i][0]表示第i天结束后为持有股票的情况下拥有的最高利润 

b状态转移方程

通过最近的一步简化问题。

分析可以得出

dp[i][0]有可能是

要想让我这一个i的值最大化,当然是要选择i-1值的最大情况 

 dp[i][1]有可能是:

c初始化

为了避免边界问题,考虑dp[0][0],和dp[0][1]即可

dp[0][0]=0

dp[0][1]=0-p[i]

d填表顺序

从左到右,两个表同时填

e返回值

max(dp[n-1][0],dp[n-1][1])

3、代码

class Solution {
public:int maxProfit(vector<int>& prices, int fee) {//建表//初始化//填表//返回值int n=prices.size();int dp[n][2];dp[0][0]=0;dp[0][1]=0-prices[0];for(int  i=1;i<n;i++){dp[i][0]=max(dp[i-1][0],dp[i-1][1]+prices[i]-fee);dp[i][1]=max(dp[i-1][0]-prices[i],dp[i-1][1]);}return max(dp[n-1][0],dp[n-1][1]);}
};

4、写题感悟

这种选择的思路感觉有点像是比较大小时,每次保存一个最小值

int my_min=0;
for(auto e:arr)
{if(my_min>e)my_min=e;
}

这道题是每次当前状态下最优解,你有更好的办法再来代替我

三、买卖股票最佳时机4

1、题目解析

2、算法原理

a状态表示方程

经验+题目要求

dp[i]粗浅的表示第i天结束后获得的利润最高

但是在第i天结束后又两种情况,手上持有股票、手上未持有股票。我们用两张表来表示:

f[i]表示第i天后持有股票时的最高利润

g[i]表示第i天后未持有股票时的最高利润

又这道题有一个附加条件,只最多只能进行k次交易。我们还有一种特殊状态需要维护、考虑,而且这种状态是影响每一天获取过程的

所以我们通过一个二维数组表示这个状态转换方程

f[i][j]表示第i天后持有股票的情况剩余j次交易机会的最高利润

g[i][j]表示第i天后未持有股票剩余j次交易机会的最高利润

b状态转移方程

还是那个小技巧:通过前一步,划分问题

最近的一步就是前一天,

持有股票的状态需要前一天未持有股票的状态买入、或者前一天本身就持有股票

未持有股票的状态需要前一天持有股票的状态卖出、或者前一天本身就未持有股票

假设这个图是表示f[i][j]的,图中三角形的位置只会由上面和右上这两种情况得出。

由此可以得出状态转移方程:

f[i][j]=max(f[i-1][j],g[i-1][j]-prices[i])

g[i][j]=max(g[i-1][j],g[i-1][j+1]+prices[i])

c初始化

初始化需要解决的问题就是边界问题。

由状态转移方程可以观察出,容易出现边界问题的下标为i-1,j+1这两个地方。

所以我们需要对第一行和最后一列特殊处理。

for(int i=0;i<=k;i++){//对第一行进行特殊处理f[0][i]=0-prices[0];g[0][i]=0;}

 持有股票就减去价格,未持有就i为0

有同学可能会疑惑,为什么f[0][0]也可以这样赋值,应为它影响不到后续的结果,这样还使代码更加简洁。

最后一列我们进过特殊情况处理即可,在填写表格时:

        for(int i=1;i<n;i++){for(int j=0;j<=k;j++){if(j==k){f[i][j]=max(f[i-1][j],g[i-1][j]-prices[i]);g[i][j]=g[i-1][j];continue;}f[i][j]=max(f[i-1][j],g[i-1][j]-prices[i]);g[i][j]=max(f[i-1][j+1]+prices[i],g[i-1][j]);}}

d填表顺序

从左到右,同时填写两个表

e返回值

返回最后一天各个情况的最大值

        int Max=INT_MIN;for(int i=0;i<=k;i++){int tem;tem=max(f[n-1][i],g[n-1][i]);Max=max(tem,Max);}return Max;

3、代码

class Solution {
public:int maxProfit(int k, vector<int>& prices) {//建表//初始化//填表//返回值int n=prices.size();int f[n][k+1],g[n][k+1];f[0][0]=0;g[0][0]=0;for(int i=1;i<=k;i++){f[0][i]=0-prices[0];g[0][i]=0;}for(int i=1;i<n;i++){for(int j=0;j<=k;j++){if(j==k){f[i][j]=max(f[i-1][j],g[i-1][j]-prices[i]);g[i][j]=g[i-1][j];continue;}f[i][j]=max(f[i-1][j],g[i-1][j]-prices[i]);g[i][j]=max(f[i-1][j+1]+prices[i],g[i-1][j]);}}int Max=INT_MIN;for(int i=0;i<=k;i++){int tem;tem=max(f[n-1][i],g[n-1][i]);Max=max(tem,Max);}return Max;}
};

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

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

相关文章

LRU的原理与实现(java)

介绍 LRU的英文全称为Least Recently Used&#xff0c;即最近最少使用。它是一种内存数据淘汰算法&#xff0c;当添加想要添加数据而内存不足时&#xff0c;它会优先将最近一段时间内使用最少的数据淘汰掉&#xff0c;再将数据添加进来。 原理 LRU的原理在介绍中就已经基本说…

继电器线圈两端为什么要反向并联二极管

原理描述&#xff1a; 电感的特点&#xff1a;穿过电感的电流不会突然变化&#xff0c;也就是说变化的电流在电感上面会产生感应电动势。 感应电动势的作用&#xff1a;阻碍电流的变化。 三极管断开之前&#xff0c;电流是穿过线圈&#xff0c;原来的电流是从上往下流动&#…

【C++】探索C++中的类与对象(下)---深入理解C++中的关键概念与应用

​​ &#x1f331;博客主页&#xff1a;青竹雾色间. &#x1f618;博客制作不易欢迎各位&#x1f44d;点赞⭐收藏➕关注 ✨人生如寄&#xff0c;多忧何为 ✨ 在C编程中&#xff0c;有许多重要的概念和特性&#xff0c;包括构造函数、explicit关键字、静态成员、友元以及内部类…

【黑马头条】-day05延迟队列文章发布审核-Redis-zSet实现延迟队列-Feign远程调用

文章目录 昨日回顾今日内容1 延迟任务1.1 概述1.2 技术对比1.2.1 DelayQueue1.2.2 RabbitMQ1.2.3 Redis实现1.2.4 总结 2 redis实现延迟任务2.0 实现思路2.1 思考2.2 初步配置实现2.2.1 导入heima-leadnews-schedule模块2.2.2 在Nacos注册配置管理leadnews-schedule2.2.3 导入表…

【Leetcode】1. 两数之和

文章目录 题目思路代码复杂度分析时间复杂度空间复杂度 结果总结 题目 题目链接&#x1f517; 给定一个整数数组 n u m s nums nums 和一个整数目标值 t a r g e t target target&#xff0c;请你在该数组中找出 和为目标值 t a r g e t target target 的那 两个 整数&…

序列超图的下一项推荐 笔记

1 Title Next-item Recommendation with Sequential Hypergraphs&#xff08;Jianling Wang、Kaize Ding、Liangjie Hong、Huan Liu、James Caverlee&#xff09;【SIGIR 2020】 2 Conclusion This study explores the dynamic meaning of items in realworld scenarios and p…

JAVA IO流学习

File类&#xff1a; File类是java.io包中很重要的一个类 File类的对象可以代表一个文件或者目录&#xff0c;可以修改文件大小、文件最后修改日期、文件名等 File对象不能操作文件的具体数据&#xff0c;即不能对文件进行读和写的操作 File的构造方法&#xff1a; File&…

并查集学习(836. 合并集合 + 837. 连通块中点的数量)

//得先加集合个数再合并&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 核心代码&#xff1a; int find(int x){//返回父节点if(x ! p[x]) {p[x] find(p[x]);//路径压缩 } //孩子不等于爸爸&#xff0c;就…

Pytorch转onnx

pytorch 转 onnx 模型需要函数 torch.onnx.export。 def export(model: Union[torch.nn.Module, torch.jit.ScriptModule, torch.jit.ScriptFunction],args: Union[Tuple[Any, ...], torch.Tensor],f: Union[str, io.BytesIO],export_params: bool True,verbose: bool False…

【QT+QGIS跨平台编译】056:【pdal_json_schema+Qt跨平台编译】(一套代码、一套框架,跨平台编译)

点击查看专栏目录 文章目录 一、pdal_json_schema介绍二、pdal下载三、文件分析四、pro文件五、编译实践一、pdal_json_schema介绍 pdal_json_schema 是与 PDAL(Point Data Abstraction Library)相关的 JSON 模式文件。PDAL 是一个用于处理和分析点云数据的开源库。JSON 模式…

DHCP-PXE

Dynamic Host Configuration Protocol 动态主机配置协议 1.Selinux 调试为Permission 防火墙配置 搭建DHCP的主机必须有一个静态地址&#xff0c;提前配置好 安装DHCP软件 服务名为dhcpd DHCP地址分配四次会话&#xff0c; DISCOVERY发现 OFFER 提供 REQUEST 回应 A…

5G网络架构及技术(二):OFDM一

ToDo: 等把这些讲义看完后得单开一个文章整理思维导图   该部分由于内容比较重要&#xff0c;OFDM是5G物理层的基础&#xff0c;但学习时直接跳到5G OFDM去看它的那些参数设置感觉没什么意义&#xff0c;还得从发展的角度进行学习&#xff0c;先从最先用到OFDM的WiFi协议开始…