剑指offer(C++)-JZ63:买卖股票的最好时机(一)(算法-动态规划)

简介: 剑指offer(C++)-JZ63:买卖股票的最好时机(一)(算法-动态规划)

题目描述:

假设你有一个数组prices,长度为n,其中prices[i]是股票在第i天的价格,请根据这个价格数组,返回买卖股票能获得的最大收益


1.你可以买入一次股票和卖出一次股票,并非每天都可以买入或卖出一次,总共只能买入和卖出一次,且买入必须在卖出的前面的某一天


2.如果不能获取到任何利润,请返回0


3.假设买入卖出均无手续费


数据范围:0≤n≤10^5,0≤val≤10^4


要求:空间复杂度 O(1),时间复杂度 O(n)

示例:

输入:

[8,9,2,5,4,7,1]


返回值:

5


说明:

在第3天(股票价格 = 2)的时候买入,在第6天(股票价格 = 7)的时候卖出,最大利润 = 7-2 = 5 ,不能选择在第2天买入,第3天卖出,这样就亏损7了;同时,你也不能在买入前卖出股票。

解题思路:

本题是动态规划的经典题目。解题思路如下:


1.定义一个多行两列的数组。第一列存储,截止到第i天时股票不持股能达到的最大收益;第二列存储,截止到第i天时股票持股能保持的最大收益。


2.第一天无法卖出,因为没有持股,所以dp[0][0]自然是0,若第一天买入,则收益就是负当天的股价。


3.从第二天开始到最后一天,循环刷新截止到某一天时最大的收益情况。


4.第i天不持股的最大收益由两个情况决定:一是第i-1天不持股的最大收益,如果该值较大,说明在前面已经完成了较优的交易;二是第i-1天持股的最大收益,在第i天卖出了,进入不持股状态,此时如果卖出的金额比之前买入的金额大,那么该值为正,如果该值较小,说明虽然完成了低买高卖的操作,但是收益并不如之前的某个操作。


5.第i天持股的最大收益由两个情况决定:一是第i-1天持股的最大收益,如果该值较大,说明前面已经拿到了较便宜的筹码;二是第i天股价的负值,说明在当天进行了买入。


6.题目的结果就是数组中最后一行第一列的数值,也就是最后一天不持股能达到的最大收益。

测试代码:

class Solution {
public:
    int maxProfit(vector<int>& prices) {
        int size = prices.size();
        // dp为多行两列的数组
        // 第一列表示股票不持股能达到的最大收益
        // 第二列表示股票持股能保持的最大收益
        vector<vector<int>> dp(size, vector<int>(2, 0));
        // 第一天无法卖出,因为没有持股
        dp[0][0] = 0;
        // 第一天买入股票,进入持股状态,收益为减去当天股价
        dp[0][1] = -prices[0];
        // 动态规划刷新第二天至最后一天的结果
        for(int i = 1; i < size; ++i){
            // 第i天不持股的最大收益由两个情况决定:
            // 一是第i-1天不持股的最大收益,如果该值较大,说明已经完成了较优的交易
            // 二是第i-1天持股的最大收益,在第i天卖出了,进入不持股状态,
            // 此时如果卖出的金额比之前买入的金额大,那么该值为正,后续
            // 除非有更大的买卖差值,否则该值不会刷新
            dp[i][0] = max(dp[i - 1][0], dp[i - 1][1] + prices[i]);
            // 第i天持股的最大收益由两个情况决定:
            // 一是第i-1天持股的最大收益,如果该值较大,说明前面已经拿到了较便宜的筹码
            // 二是第i天股价的负值,说明在当天进行了买入
            dp[i][1] = max(dp[i - 1][1], -prices[i]);
        }
        // 最后一天不持股能达到的最大收益就是最优结果
        return dp[size - 1][0];
    }
};


相关文章
|
2天前
|
机器学习/深度学习 存储 算法
数据结构与算法 动态规划(启发式搜索、遗传算法、强化学习待完善)
数据结构与算法 动态规划(启发式搜索、遗传算法、强化学习待完善)
13 1
|
2天前
|
存储 缓存 算法
C++从入门到精通:4.6性能优化——深入理解算法与内存优化
C++从入门到精通:4.6性能优化——深入理解算法与内存优化
|
2天前
|
存储 算法 程序员
C++从入门到精通:2.2.1标准库与STL容器算法深度解析
C++从入门到精通:2.2.1标准库与STL容器算法深度解析
|
2天前
|
算法
代码随想录算法训练营第五十六天 | LeetCode 647. 回文子串、516. 最长回文子序列、动态规划总结
代码随想录算法训练营第五十六天 | LeetCode 647. 回文子串、516. 最长回文子序列、动态规划总结
34 1
|
2天前
|
算法
算法系列--动态规划--特殊的状态表示--分析重复子问题(下)
算法系列--动态规划--特殊的状态表示--分析重复子问题(下)
16 0
|
2天前
|
算法
算法系列--动态规划--特殊的状态表示--分析重复子问题(上)
算法系列--动态规划--特殊的状态表示--分析重复子问题
18 0
|
2天前
|
算法
算法系列--动态规划--背包问题(5)--二维费用背包问题(下)
算法系列--动态规划--背包问题(5)--二维费用背包问题(下)
14 0
|
2天前
|
算法
算法系列--动态规划--背包问题(5)--二维费用背包问题(上)
算法系列--动态规划--背包问题(5)--二维费用背包问题(上)
21 0
|
2天前
|
算法
算法系列--动态规划--背包问题(4)--完全背包拓展题目(下)
算法系列--动态规划--背包问题(4)--完全背包拓展题目(下)
22 0
|
2天前
|
算法
算法系列--动态规划--背包问题(4)--完全背包拓展题目(上)
算法系列--动态规划--背包问题(4)--完全背包拓展题目(上)
20 0