< 每日算法 - JavaScript解析:一文解决 “ 买卖股票 ” 系列算法题 >

在这里插入图片描述


本篇文章涉及题目均来自 leetCode

小温建议各位小伙伴看完,立马去leetCode上手写写,毕竟俗话说的好: 好记性不如烂笔头

一、基础题目

接下来,就由小温带大家解决 “买卖股票”系列算法题, 以最基础的 121 题:买卖股票的最佳时机(简单) 为例,讲解解决思路:

> 题目

给定一个数组 prices ,它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。你只能选择 某一天 买入这只股票,并选择在 未来的某一个不同的日子 卖出该股票( 此段话表示,只允许交易一次,也就是买和卖各一次。k = 1 )。

设计一个算法来计算你所能获取的最大利润,返回你可以从这笔交易中获取的最大利润。如果你不能获取任何利润,返回 0 。

示例 1:

输入:[7,1,5,3,6,4]
输出:5
解释:在第 2 天(股票价格 = 1)的时候买入,在第 5 天(股票价格 = 6)的时候卖出,最大利润 = 6-1 = 5 。
     注意利润不能是 7-1 = 6, 因为卖出价格需要大于买入价格;同时,你不能在买入前卖出股票。

示例 2:

输入:prices = [7,6,4,3,1]
输出:0
解释:在这种情况下, 没有交易完成, 所以最大利润为 0

> 解题思路

阐述效果图

通过上图可以知道,我们需要借助已知的数据,配合定义未知的变量去推出最大利润! 而定义这个推理公式的方法,叫做 “ 动态规划 ”。需要清楚各个数据之间的练习,接下来分析一下需要定义那些已知常量未知的变量

根据题目可以知道每天的股票的波动数据,结合常识,需要定义以下内容:

定义操作

  • 买入
  • 卖出
  • 不操作

定义状态

  • i: 天数
  • k: 交易次数,每次交易包含买入和卖出,这里我们只在买入的时候需要将 k - 1
  • 0: 不持有股票时,买入费用
  • 1: 持有股票时开销,包含购入的价格,所以计算利润时,需要减去购入价格

动态规划值所需变量

  • dp[i][k][0] : 第 i 天 第 k 次交易中,手中无股票, 计算公式为:当天股票价格 - 购入股票的费用。
  • dp[i][k][1] : 第 i 天 第 k 次交易中,手中有股票,买入股票的费用。

效果图

// 今天没有持有股票,分为两种情况
// 1. dp[i - 1][k][0],昨天没有持有,今天不操作。 
// 2. dp[i - 1][k][1] + prices[i] 昨天持有,今天卖出,今天手中就没有股票了。
dp[i][k][0] = Math.max(dp[i - 1][k][0], dp[i - 1][k][1] + prices[i])


// 今天持有股票,分为两种情况:
// 1.dp[i - 1][k][1] 昨天持有,今天不操作
// 2.dp[i - 1][k - 1][0] - prices[i] 昨天没有持有,今天买入。
dp[i][k][1] = Math.max(dp[i - 1][k][1], dp[i - 1][k - 1][0] - prices[i])

//最大利润就是这俩种情况的最大值

====> 由于 此题k为1,所以转化为以下情况 <====

//第i天不持有 由 第i-1天不持有然后不操作 和 第i-1天持有然后卖出 两种情况的最大值转移过来
dp[i][1][0] = Math.max(dp[i - 1][1][0], dp[i - 1][1][1] + prices[i])

//第i天持有 由 第i-1天持有然后不操作 和 第i-1天不持有然后买入 两种情况的最大值转移过来
dp[i][1][1] = Math.max(dp[i - 1][1][1], dp[i - 1][0][0] - prices[i])
            = Math.max(dp[i - 1][1][1], -prices[i])
            // k=0时 没有交易次数,dp[i - 1][0][0] = 0

k是固定值1,不影响结果,所以可以不用管,简化之后如下:

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

完整代码

//时间复杂度O(n) 空间复杂度O(n),dp数组第二维是常数
const maxProfit = function (prices) {
    let n = prices.length;
    let dp = Array.from(new Array(n), () => new Array(2));
    dp[0][0] = 0; //第0天不持有
    dp[0][1] = -prices[0]; //第0天持有
    for (let i = 1; i < n; i++) {
        dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][1] + prices[i]);
        dp[i][1] = Math.max(dp[i - 1][1], -prices[i]);
    }
    return dp[n - 1][0];
};

状态压缩,dp[i] 只和 dp[i - 1] 有关,去掉一维

//时间复杂度O(n) 空间复杂度O(1)
const maxProfit = function (prices) {
	let n = prices.length;
	let dp = Array.from(new Array(n), () => new Array(2));
	dp[0] = 0;
	dp[1] = -prices[0];
	for (let i = 1; i < n; i++) {
	    dp[0] = Math.max(dp[0], dp[1] + prices[i]);
	    dp[1] = Math.max(dp[1], -prices[i]);
	}
	return dp[0];
};

//语意化
const maxProfit = function (prices) {
    let n = prices.length;
    let sell = 0;
    let buy = -prices[0];
    for (let i = 1; i < n; i++) {
        sell = Math.max(sell, buy + prices[i]);
        buy = Math.max(buy, -prices[i]);
    }
    return sell;
};

二、添加条件:当交易次数为 ∞ 时

> 题目

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

在每一天,你可以决定是否购买和/或出售股票。你在任何时候 最多 只能持有 一股 股票。你也可以先购买,然后在 同一天 出售。

返回 你能获得的 最大 利润 。

示例 1:

输入:prices = [7,1,5,3,6,4]
输出:7
解释:在第 2 天(股票价格 = 1)的时候买入,在第 3 天(股票价格 = 5)的时候卖出, 这笔交易所能获得利润 = 5 - 1 = 4 。
     随后,在第 4 天(股票价格 = 3)的时候买入,在第 5 天(股票价格 = 6)的时候卖出, 这笔交易所能获得利润 = 6 - 3 = 3 。
     总利润为 4 + 3 = 7

示例 2:

输入:prices = [1,2,3,4,5]
输出:4
解释:在第 1 天(股票价格 = 1)的时候买入,在第 5 天 (股票价格 = 5)的时候卖出, 这笔交易所能获得利润 = 5 - 1 = 4 。
     总利润为 4

示例 3:

输入:prices = [7,6,4,3,1]
输出:0
解释:在这种情况下, 交易无法获得正利润,所以不参与交易可以获得最大利润,最大利润为 0

> 解决思路

股票对应的题目,只要能理解第一题的动态规划公式,能理解推论,基本上大致都相同。在添加交易次数为无穷的时候,只需要在下一次买入时,加上上次交易剩余的利润即可,在 k 同样不影响结果,简化之后如下:

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])

语意化之后的核心代码如下:

sell = Math.max(sell, buy + prices[i])
buy = Math.max(buy, sell - prices[i])

三、添加条件:当交易次数为 K = number 时

> 题目

给定一个整数数组 prices ,它的第 i 个元素 prices[i] 是一支给定的股票在第 i 天的价格,和一个整型 k 。

设计一个算法来计算你所能获取的最大利润。你最多可以完成 k 笔交易。也就是说,你最多可以买 k 次,卖 k 次

注意:你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。

示例 1:

输入:k = 2, prices = [2,4,1]
输出:2
解释:在第 1  (股票价格 = 2) 的时候买入,在第 2  (股票价格 = 4) 的时候卖出,这笔交易所能获得利润 = 4-2 = 2

示例 2:

输入:k = 2, prices = [3,2,6,5,0,3]
输出:7
解释:在第 2  (股票价格 = 2) 的时候买入,在第 3  (股票价格 = 6) 的时候卖出, 这笔交易所能获得利润 = 6-2 = 4 。
     随后,在第 5  (股票价格 = 0) 的时候买入,在第 6  (股票价格 = 3) 的时候卖出, 这笔交易所能获得利润 = 3-0 = 3

> 解决思路

根据题目可以得出,我们不能只基于股票数据数组来循环,需要将每个数据假设它交易 K 次,取所有可能的最大利润值!

所以在我们运算时,需要借助 二维数组 或者是 对象数组 来进行缓存,缓存上次最优解的购入和售出的利润! 在当次计算时,加上上次交易带来的利润进行运算。

具体逻辑如下:

/**
 * @param {number} k
 * @param {number[]} prices
 * @return {number}
 */
var maxProfit = function(k, prices) {
    let len = prices.length
	// 定义缓存变量,缓存每次交易的购入和售出费用
    let pricesList = new Array(k)
    for(let j = 0; j <= k; j++) {
        pricesList[j] = {
            buy: -prices[0],
            sell: 0
        }
    }



    for(let i = 0; i < len; i++) {
        for(let j = 1; j <= k; j++) {
            pricesList[j].buy = Math.max(pricesList[j].buy, pricesList[j - 1].sell - prices[i])
            pricesList[j].sell = Math.max(pricesList[j].sell, pricesList[j].buy + prices[i])
        }
    }
    return pricesList[k].sell
};

本质还是和上一题,交易次数为无穷的时差不多的核心代码,属于是换汤不换药了! 只需要在本次计算时,加上上次剩余的利润即可!

四、添加条件: 每次交易时,扣除 fee 数额的手续费

这题是122题的变种,我们需要知道,当每次交易卖出时,为该次交易结束。在交易结束时,对总利润减去本次交易的手续费即可!

核心代码如下:

buy = Math.max(buy, sell - prices[i])
sell = Math.max(sell, buy + prices[i] - fee)

👉 结论

通过本篇文章的讲述,相信大家基本上能搞懂 “ 买卖股票 ” 系列的动态规划公式了,万变不离其宗! 只要能理解:买入和卖出 及 当天股票价格之间的关系,基本上都能推出了!好了,感谢各位小伙伴读到这里,如果感觉对你有所帮助,请勿吝惜你的小手指呀!给小温一个三连吧!

往期内容 💨

🔥 < 每日算法:一文带你认识 “ 双指针算法 ” >

🔥 < 每日小技巧: 基于Vue状态的过渡动画 - Transition 和 TransitionGroup>

🔥 < JavaScript技术分享: 大文件切片上传 及 断点续传思路 >

🔥 < 每日份知识快餐:axios是什么?如何在Vue中 封装 axios ? >

🔥 < 面试知识点:什么是 Node.js ?有哪些优缺点?应用场景? >

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
很不错的抓包工具 捕包选项 1、选择好捕包网卡,左连还有一些其它捕包条件供选择,如果当所选网卡不支持“杂项接收”功能,系统会提示相应信息,出现该情况时您将无法获取与本网卡无关的数据包,换言之,您无法获取其他电脑之间的通讯包,所以, 建议您更换网卡。 不支持“杂项接收”的网卡,多数为一部分无线网卡及少数专用服务器/笔记本网卡。 2、协议过滤 通常情况下,可不选,除非您对协议类型较为熟悉。 3、设置捕包缓冲 确省的捕包缓冲区大小为 1M,如果您的要追踪的网络规模较大,可适当调大该值;另外,如果追踪主机 CPU 处理能力不够,也需加大缓冲;否则,可能出现丢包的情况。 4、IP过滤 IP过滤里可以设置想要捕包的IP地址或是设置要排除的IP地址等信息。 5、端口过滤 端口过滤过滤里可以设置想要捕包的端口或是设置要排除过滤的端口等信息。 捕包分析 1、设置捕包过滤项 这里的过滤和“追踪任务”过滤设置是独立分开的,请不要混淆,其可选内容项更多。 上述选项中,最为复杂的是“数据块匹配”部分,详细的介绍将在下面的章节部分出现,这里只需要配置好正确的网卡即可,其他选项可以不做任何设置。 IP包回放 1、有助于了解原始包通讯的地理分布情况。 2、通过将IP包回放到网卡上,模拟原始IP包在网络上传输情况,也可供同类捕包软件捕获分析。 通讯协议分析 捕包准备 捕包分析工具条: 开始捕包前,用户需先进行过滤设置,选项内容包括: 选网卡 如果您有多块网卡,需要选中能捕包到预想中的数据的网卡。 协议过滤 针对Internet通讯部分,常见的IP包类型为:TCP/UDP/ICMP。绝大部分是TCP连接的,比如HTTP(s)/SMTP/POP3/FTP/TELNET等等;一部分聊天软件中除了采用TCP通讯方式外,也采用了UDP的传输方式,如QQ/SKYPE等;而常见的ICMP包是由客户的Ping产生的。 IP过滤 “IP过滤”在捕包过滤使用最为常见,IP匹配主要分两类:一是不带通讯方向,单纯的是范围的匹配,如上图中的“From:to”类型;另外一类是带通讯方向的一对一匹配,如上图“”类型,不仅匹配IP地址,也匹配通讯的源IP和目标IP的方向。 端口过滤 “端口过滤”只针对两种类型的DoD-IP包:TCP/UDP。 数据区大小 “数据区大小” 的匹配针对所有DoD-IP类型包,不过需要说明的是,TCP/UDP的IP数据区是以实际数据区位置开始计算的,而其他类型的则把紧随IP包头后面的部分当作数据区。 数据块匹配 “数据块匹配”较为复杂,但却非常有用. 在这里,用户可以输入文本,也可以输入二进制,可以选择特定位置的匹配,也可以选择任意位置的匹配,总之,该设置非常灵活好用。 结束条件 缺省条件下,当捕获的包占用空间多余10M时,自动停止。 结束于某个时间点,是指捕包的截止时间。 分析捕获包 用户按下“开始”按钮启动捕包功能后,列表框中会自动显示出符合条件的数据包,并附带简单的解析。选中“分析”左边和右下部分是分析结果,右上部是原始二进制代码,选中左边某一条目时,在右边二进制区域的色块和其一一对应。 如果用户以前少有接触协议分析部分,Tracknet可以很好地协助您深入了解TCP/IP协议。我们将在网站定期提供一些捕获样例包,协助用户学习分析各种类型的IP包。
以下是使用C++实现会议安排问的贪心算法示例代码: ```cpp #include <iostream> #include <vector> #include <algorithm> // 定义会议结构体 struct Meeting { int start; int end; }; // 比较函数,按照会议结束时间升序排序 bool compare(Meeting m1, Meeting m2) { return m1.end < m2.end; } // 贪心算法实现会议安排 std::vector<Meeting> scheduleMeetings(std::vector<Meeting>& meetings) { // 按照结束时间排序会议 std::sort(meetings.begin(), meetings.end(), compare); std::vector<Meeting> schedule; // 会议安排列表 schedule.push_back(meetings[0]); // 将第一个会议加入安排列表 for (int i = 1; i < meetings.size(); i++) { // 如果当前会议与已安排的最后一个会议不冲突,则将该会议加入安排列表 if (meetings[i].start >= schedule.back().end) { schedule.push_back(meetings[i]); } } return schedule; } int main() { std::vector<Meeting> meetings = {{1, 3}, {2, 4}, {3, 6}, {5, 7}, {8, 9}, {9, 10}}; std::vector<Meeting> schedule = scheduleMeetings(meetings); // 输出会议安排 for (int i = 0; i < schedule.size(); i++) { std::cout << "Meeting " << i+1 << ": " << schedule[i].start << "-" << schedule[i].end << std::endl; } return 0; } ``` 以上代码使用了结构体 `Meeting` 来表示一个会议,其中包含开始时间和结束时间。首先,通过定义一个比较函数 `compare` 来按照会议结束时间进行排序。然后,通过 `scheduleMeetings` 函数来实现会议安排的贪心算法。最后,在 `main` 函数中调用 `scheduleMeetings` 函数并输出会议安排结果。 运行以上代码,会输出以下结果: ``` Meeting 1: 1-3 Meeting 2: 5-7 Meeting 3: 8-9 ``` 这表示在给定的会议列表中,按照贪心算法安排的会议有三个:第一个会议从1到3,第二个会议从5到7,第三个会议从8到9。这些会议之间不存在时间冲突,并且结束时间最早。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

技术宅小温

你小小的鼓励,是我搬砖的动力!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值