-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathProblem_0309_maxProfit.cc
130 lines (122 loc) · 5.24 KB
/
Problem_0309_maxProfit.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
#include <iostream>
#include <vector>
#include "UnitTest.h"
using namespace std;
class Solution
{
public:
// 暴力尝试
// buy == false 目前可以交易,而且当前没有购买行为
// buy == true 已经买了,买入价buyPrices,待卖出
int f(vector<int>& prices, bool buy, int index, int buyPrice)
{
if (index >= prices.size())
{
return 0;
}
if (buy)
{
// 之前买过,那么本次可以选择不出售,出售
int noSell = f(prices, true, index + 1, buyPrice);
// 因为售出股票后有 1 天的冷冻期,所以 index + 2
int yesSell = prices[index] - buyPrice + f(prices, false, index + 2, 0);
return std::max(noSell, yesSell);
}
else
{
// 之前没买过,那么本次可以选择购买、或不购买
int noBuy = f(prices, false, index + 1, 0);
int yesBuy = f(prices, true, index + 1, prices[index]);
return std::max(noBuy, yesBuy);
}
}
int maxProfit1(vector<int>& prices) { return f(prices, false, 0, 0); }
// 最优尝试如下:
// buy[i] : 在0...i范围上,最后一次操作是buy动作,
// 这最后一次操作有可能发生在i位置,也可能发生在i之前
// buy[i]值的含义是:max{ 所有可能性[之前交易获得的最大收益 - 最后buy动作的收购价格] }
// 比如:arr[0...i]假设为[1,3,4,6,2,7,1...i之后的数字不用管]
// 什么叫,所有可能性[之前交易获得的最大收益 - 最后buy动作的收购价格]?
// 比如其中一种可能性:
// 假设最后一次buy动作发生在2这个数的时候,那么之前的交易只能在[1,3,4]上结束,因为6要cooldown的,
// 此时最大收益是多少呢?是4-1==3。那么,之前交易获得的最大收益 - 最后buy动作的收购价格 = 3 - 2 =
// 1 另一种可能性:
// 再比如最后一次buy动作发生在最后的1这个数的时候,那么之前的交易只能在[1,3,4,6,2]上发生,因为7要cooldown的,
// 此时最大收益是多少呢?是6-1==5。那么,之前交易获得的最大收益 - 最后buy动作的收购价格 = 5 - 1 =
// 4 除了上面两种可能性之外,还有很多可能性,你可以假设每个数字都是最后buy动作的时候,
// 所有可能性中,(之前交易获得的最大收益 - 最后buy动作的收购价格)的最大值,就是buy[i]的含义
// 为啥buy[i]要算之前的最大收益 - 最后一次收购价格?尤其是最后为什么要减这么一下?
// 因为这样一来,当你之后以X价格做成一笔交易的时候,当前最好的总收益直接就是 X + buy[i]了
//
// sell[i] :0...i范围上,最后一次操作是sell动作,这最后一次操作有可能发生在i位置,也可能发生在之前
// sell[i]值的含义:0...i范围上,最后一次动作是sell的情况下,最好的收益
//
// 于是通过分析,能得到以下的转移方程:
// buy[i] = Math.max(buy[i - 1], sell[i - 2] - prices[i])
// 如果i位置没有发生buy行为,说明有没有i位置都一样,那么buy[i] = buy[i-1],这显而易见
// 如果i位置发生了buy行为, 那么buy[i] = sell[i - 2] - prices[i],
// 因为你想在i位置买的话,你必须保证之前交易行为发生在0...i-2上,
// 因为如果i-1位置有可能参与交易的话,i位置就要cooldown了,
// 而且之前交易行为必须以sell结束,你才能buy,而且要保证之前交易尽可能得到最好的利润,
// 这正好是sell[i - 2]所代表的含义,并且根据buy[i]的定义,最后一定要 - prices[i]
//
// sell[i] = Math.max(sell[i - 1], buy[i - 1] + prices[i])
// 如果i位置没有发生sell行为,那么sell[i] = sell[i-1],这显而易见
// 如果i位置发生了sell行为,那么我们一定要找到 {之前获得尽可能好的收益 -
// 最后一次的收购价格尽可能低}, 而这正好是buy[i - 1]的含义!之前所有的"尽可能"中,最好的一个!
int maxProfit2(vector<int>& prices)
{
if (prices.size() < 2)
{
return 0;
}
int N = prices.size();
vector<int> buy(N);
vector<int> sell(N);
buy[1] = std::max(-prices[0], -prices[1]);
sell[1] = std::max(0, prices[1] - prices[0]);
for (int i = 2; i < N; i++)
{
buy[i] = std::max(buy[i - 1], sell[i - 2] - prices[i]);
sell[i] = std::max(sell[i - 1], buy[i - 1] + prices[i]);
}
return sell[N - 1];
}
// 方法2的空间优化
int maxProfit3(vector<int>& prices)
{
if (prices.size() < 2)
{
return 0;
}
int buy1 = std::max(-prices[0], -prices[1]);
int sell1 = std::max(0, prices[1] - prices[0]);
int sell2 = 0;
for (int i = 2; i < prices.size(); i++)
{
int tmp = sell1;
sell1 = std::max(sell1, buy1 + prices[i]);
buy1 = std::max(buy1, sell2 - prices[i]);
sell2 = tmp;
}
return sell1;
}
};
void testMaxProfit()
{
Solution s;
vector<int> p1 = {1, 2, 3, 0, 2};
vector<int> p2 = {1};
EXPECT_EQ_INT(3, s.maxProfit1(p1));
EXPECT_EQ_INT(0, s.maxProfit1(p2));
EXPECT_EQ_INT(3, s.maxProfit2(p1));
EXPECT_EQ_INT(0, s.maxProfit2(p2));
EXPECT_EQ_INT(3, s.maxProfit3(p1));
EXPECT_EQ_INT(0, s.maxProfit3(p2));
EXPECT_SUMMARY;
}
int main()
{
testMaxProfit();
return 0;
}