买卖股票——从LeetCode题海中总结常见套路

2024-03-03 23:40

本文主要是介绍买卖股票——从LeetCode题海中总结常见套路,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这是一类很著名很经典的问题,经常在字节等面试中看见,对深入理解记忆化搜索、动态规划、状态机等技巧非常有帮助,写一篇文章来细细玩味这类问题!

目录

经典记忆化搜索:LeetCode121.买卖股票的最佳时机

两次遍历,一次找最小值,一次找最大值:

时间上进行优化:

空间上进行优化:

类似的记忆化搜索:剑指offer63.股票的最大利润

意料之外的贪心:LeetCode122.买卖股票的最佳时机II

DP状态机:LeetCode123.买卖股票的最佳时机III

状压DP:LeetCode188.买卖股票的最佳时机IV


经典记忆化搜索:LeetCode121.买卖股票的最佳时机

这是极其经典的记忆化搜索!想想什么任意一个时刻卖出股票最多的钱是多少?是当前的价格减去前面最小的那次

所以依照这个思路,定义一个数组存储前面最小的那次,然后计算出当前卖出股票所挣的钱

挣钱最多的就是最佳时机!

这里给出的依旧是一个O(n*n)的暴力思路,因为这里当前状态和往日状态息息相关,所以就为了记忆化搜索提供了切入点

具体看代码:

两次遍历,一次找最小值,一次找最大值:

class Solution {
public:int maxProfit(vector<int>& prices) {if(prices.size()<=1)return 0;int minnum[prices.size()];minnum[0] = minnum[1] = prices[0];for(int i = 1;i<prices.size();i++){minnum[i] = min(minnum[i-1],prices[i-1]);}int ans = 0;for(int i = prices.size()-1;i>0;i--){if((prices[i]-minnum[i])>ans)ans = prices[i]-minnum[i];}return ans;}
};

时间上进行优化:

将两次遍历改成以此遍历:

class Solution {
public:int maxProfit(vector<int>& prices) {if (prices.size()<=1)return 0;// minnum[i]表示前i个元素中最小的元素(不包括本身)int minnum[prices.size()];minnum[0] = prices[0];int ans = 0;for (int i = 1; i<prices.size(); i++) {// 记忆化搜索找出前i个元素中最小的元素minnum[i] = min(minnum[i-1], prices[i-1]);// 最大的利润等于当前的价格减去前i个元素中最小的ans = max((prices[i]-minnum[i]), ans);}return ans;}
};

空间上进行优化:

用minnum代替minnum[prices.size()],可以降低空间复杂度

class Solution {
public:int maxProfit(vector<int>& prices) {if (prices.size()<=1)return 0;// minnum表示前i个元素中最小的元素(不包括本身)int minnum = prices[0];int ans = 0;for (int i = 1; i<prices.size(); i++) {// 记忆化搜索找出前i个元素中最小的元素minnum = min(minnum, prices[i-1]);// 最大的利润等于当前的价格减去前i个元素中最小的ans = max(prices[i]-minnum, ans);}return ans;}
};

类似的记忆化搜索:剑指offer63.股票的最大利润

其实几乎是一模一样的题呀

 

class Solution {
public:int maxProfit(vector<int>& prices) {// 类似LeetCode121.买卖股票的最佳时机if (prices.size()<2)return 0;int minnum = prices[0];int ans = 0;for (int i = 1; i<prices.size(); i++) {minnum = min(prices[i-1], minnum);ans = max(ans, prices[i]-minnum);}return ans;}
};

意料之外的贪心:LeetCode122.买卖股票的最佳时机II

这道题确实出乎意料,表面上是前面那题改了一下,从只能买卖一次改成了可以买卖很多次

这里题目中一个关键的提示:尽可能多的完成更多的交易

这个提示就给了贪心的机会了!贪心策略:只要今天比昨天的大,就卖出,追求局部最优解,其实这就是全局最优解(不证明)

class Solution {
public:int maxProfit(vector<int>& prices) {// 贪心策略:只要今天比昨天的大,就卖出,追求局部最优解// 题目有提示:你可以尽可能地完成更多的交易(多次买卖一支股票)。int ans = 0;for (int i = 1; i<prices.size(); i++) {if ((prices[i]-prices[i-1])>0) {ans += (prices[i] - prices[i-1]);}}return ans;}
};

DP状态机:LeetCode123.买卖股票的最佳时机III

class Solution {
public:int maxProfit(vector<int>& prices) {/*对于任意一天考虑四个变量:fstBuy: 在该天第一次买入股票可获得的最大收益 fstSell: 在该天第一次卖出股票可获得的最大收益secBuy: 在该天第二次买入股票可获得的最大收益secSell: 在该天第二次卖出股票可获得的最大收益分别对四个变量进行相应的更新, 最后secSell就是最大收益值(secSell >= fstSell)*/int INF = (1<<21);int fstBuy = -INF, fstSell = 0;int secBuy = -INF, secSell = 0;for (auto p : prices) {fstBuy = max(fstBuy, -p);fstSell = max(fstSell, fstBuy + p);secBuy = max(secBuy, fstSell - p);secSell = max(secSell, secBuy + p); }return secSell;}
};

 

状压DP:LeetCode188.买卖股票的最佳时机IV

真TM是太难了。。。。

 

 

 

 

状压DP的坑还没填,留着后面搞吧。。。

这篇关于买卖股票——从LeetCode题海中总结常见套路的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Qt实现网络数据解析的方法总结

《Qt实现网络数据解析的方法总结》在Qt中解析网络数据通常涉及接收原始字节流,并将其转换为有意义的应用层数据,这篇文章为大家介绍了详细步骤和示例,感兴趣的小伙伴可以了解下... 目录1. 网络数据接收2. 缓冲区管理(处理粘包/拆包)3. 常见数据格式解析3.1 jsON解析3.2 XML解析3.3 自定义

Java Spring 中 @PostConstruct 注解使用原理及常见场景

《JavaSpring中@PostConstruct注解使用原理及常见场景》在JavaSpring中,@PostConstruct注解是一个非常实用的功能,它允许开发者在Spring容器完全初... 目录一、@PostConstruct 注解概述二、@PostConstruct 注解的基本使用2.1 基本代

SQL中redo log 刷⼊磁盘的常见方法

《SQL中redolog刷⼊磁盘的常见方法》本文主要介绍了SQL中redolog刷⼊磁盘的常见方法,将redolog刷入磁盘的方法确保了数据的持久性和一致性,下面就来具体介绍一下,感兴趣的可以了解... 目录Redo Log 刷入磁盘的方法Redo Log 刷入磁盘的过程代码示例(伪代码)在数据库系统中,r

Python实现图片分割的多种方法总结

《Python实现图片分割的多种方法总结》图片分割是图像处理中的一个重要任务,它的目标是将图像划分为多个区域或者对象,本文为大家整理了一些常用的分割方法,大家可以根据需求自行选择... 目录1. 基于传统图像处理的分割方法(1) 使用固定阈值分割图片(2) 自适应阈值分割(3) 使用图像边缘检测分割(4)

Windows Docker端口占用错误及解决方案总结

《WindowsDocker端口占用错误及解决方案总结》在Windows环境下使用Docker容器时,端口占用错误是开发和运维中常见且棘手的问题,本文将深入剖析该问题的成因,介绍如何通过查看端口分配... 目录引言Windows docker 端口占用错误及解决方案汇总端口冲突形成原因解析诊断当前端口情况解

SQL BETWEEN 的常见用法小结

《SQLBETWEEN的常见用法小结》BETWEEN操作符是SQL中非常有用的工具,它允许你快速选取某个范围内的值,本文给大家介绍SQLBETWEEN的常见用法,感兴趣的朋友一起看看吧... 在SQL中,BETWEEN是一个操作符,用于选取介于两个值之间的数据。它包含这两个边界值。BETWEEN操作符常用

python中各种常见文件的读写操作与类型转换详细指南

《python中各种常见文件的读写操作与类型转换详细指南》这篇文章主要为大家详细介绍了python中各种常见文件(txt,xls,csv,sql,二进制文件)的读写操作与类型转换,感兴趣的小伙伴可以跟... 目录1.文件txt读写标准用法1.1写入文件1.2读取文件2. 二进制文件读取3. 大文件读取3.1

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

前端下载文件时如何后端返回的文件流一些常见方法

《前端下载文件时如何后端返回的文件流一些常见方法》:本文主要介绍前端下载文件时如何后端返回的文件流一些常见方法,包括使用Blob和URL.createObjectURL创建下载链接,以及处理带有C... 目录1. 使用 Blob 和 URL.createObjectURL 创建下载链接例子:使用 Blob

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve