Let's take a look at an example. Assume that you somehow have the ability to predict stock market for the next N days. Say you can trade stock exactly once a day, and the total number of trades you can make within the next N days is constrained to some number, say n. In addition, you can not own more than 1 stock at a time.
So, basically, given stock prices for next N consecutive days, [x1, x2, ..., xN], and given maximum of n total trades all on different days, when will you buy and sell each stock so that you can maximize the profit? Note that you must buy-sell-buy-sell and so forth with no consecutive buys or sells.
For example, if the next 5 day market price will be [5, 2, 2, 5, 3], and if I have at most 2 buy-sell trades to do, then I would buy on day 2 or 3 and sell on day 4, leaving me with +3 in profit. I will buy-sell only once, since this maximizes the profit.
It is easy when N is small, but it gets very difficult as N grows large. Given N = 50 with prices given by [8, 44, 39, 4, 41, 0, 19, 29, 44, 43, 44, 21, 34, 39, 23, 7, 1, 24, 30, 34, 40, 26, 28, 11, 44, 19, 48, 12, 5, 10, 2, 21, 0, 24, 8, 44, 36, 20, 6, 2, 4, 21, 14, 6, 47, 31, 23, 29, 4, 36] and you have at most 10 buy-sell trades allowed. What is your maximum profit?
To tackle this problem, I would first break down the problem. Here, it is asking for the maximum profit for at most n stock buy-sell stock trades. I will split it into sub-problems where I can find the maximum profit for exactly n buy-sell trades, and later I can search for the max among them.
So, the sub-problem is now to find the maximum profit given N-day forecast where I need to trade exactly n times. Since each trade is either buy or sell, I will once again split this sub-problem into sub-sub-problem as follows:
B(x, N, n): given N-day forecast of prices x, find the maximum cash after exactly n-1 buy-sell trades and another buy by the Nth day.
S(x, N, n): given N-day forecast of prices x, find the maximum cash after exactly n buy-sell trades by the Nth day.
With these functions, it is not too difficult to come up with a simple recursion relationship. Perhaps it is easy to see this from Python code below:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
def maxCashAfterBuy(x, k, n): | |
""" | |
x: a list of daily stock price [x1, x2, x3, ...] | |
k: constraint on trade days from the beginnig, | |
i.e., [x1, x2, ..., xk] | |
n: total number of buys to consider | |
return maximum possible cash after buying and selling stock | |
n and n-1 times, respectively in the interval [x1, x2, ..., xk] | |
no stock can be bought and sold on the same day | |
return -inf if given condition is not possible | |
""" | |
# not possible condition | |
if k > len(x) or n < 0 or k < 0 or len(x) <= 0 or n > (k+1)/2: | |
return -float('inf') | |
# initial condition: start with no cash | |
if n == 0: | |
return 0 | |
# just bought the stock on the first day | |
if k == 1 and n == 1: | |
return -x[0] | |
# recursive | |
return max(maxCashAfterBuy(x, k-1, n), | |
maxCashAfterSell(x, k-1, n-1) - x[k-1]) | |
def maxCashAfterSell(x, k, n): | |
""" | |
x: a list of daily stock price [x1, x2, x3, ...] | |
k: constraint on trade days from the beginnig, | |
i.e., [x1, x2, ..., xk] | |
n: total number of sells to consider | |
return maximum possible cash after buying and selling stock | |
n times each in the interval [x1, x2, ..., xk] | |
no stock can be bought and sold on the same day | |
return -inf if given condition is not possible | |
""" | |
# not possible condition | |
if k > len(x) or n < 0 or k < 0 or len(x) <= 0 or n > k/2: | |
return -float('inf') | |
# initial condition: start with no cash | |
if n == 0: | |
return 0 | |
# just bought the stock on the first day and sold on the second day | |
if k == 2 and n == 1: | |
return x[1] - x[0] | |
# recursive | |
return max(maxCashAfterSell(x, k-1, n), | |
maxCashAfterBuy(x, k-1, n) + x[k-1]) |
It is probably better to explain in code below:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
def maxCashAfterBuy(x, k, n, prevB, prevS): | |
""" | |
x: a list of daily stock price [x1, x2, x3, ...] | |
k: constraint on trade days from the beginnig, | |
i.e., [x1, x2, ..., xk] n: total number of buys to consider | |
prevB: list of values for maxCashAfterBuy(x, k-1, n) for n >= 0 | |
prevS: list of values for maxCashAfterSell(x, k-1, n) for n >= 0 | |
return maximum possible cash after buying and selling stock | |
n and n-1 times, respectively in the interval [x1, x2, ..., xk] | |
no stock can be bought and sold on the same day | |
return -inf if given condition is not possible | |
""" | |
# not possible condition | |
if k > len(x) or n < 0 or k < 0 or len(x) <= 0 or n > (k+1)/2: | |
return -float('inf') | |
# initial condition: start with no cash | |
if n == 0: | |
return 0 | |
# just bought the stock on the first day | |
if k == 1 and n == 1: | |
return -x[0] | |
# recursive | |
return max(prevB[n], prevS[n-1] - x[k-1]) | |
def maxCashAfterSell(x, k, n, prevB, prevS): | |
""" | |
x: a list of daily stock price [x1, x2, x3, ...] | |
k: constraint on trade days from the beginnig, | |
i.e., [x1, x2, ..., xk] | |
n: total number of sells to consider | |
prevB: list of values for maxCashAfterBuy(x, k-1, n) for n >= 0 | |
prevS: list of values for maxCashAfterSell(x, k-1, n) for n >= 0 | |
return maximum possible cash after buying and selling stock | |
n times each in the interval [x1, x2, ..., xk] | |
no stock can be bought and sold on the same day | |
return -inf if given condition is not possible | |
""" | |
# not possible condition | |
if k > len(x) or n < 0 or k < 0 or len(x) <= 0 or n > k/2: | |
return -float('inf') | |
# initial condition: start with no cash | |
if n == 0: | |
return 0 | |
# just bought the stock on the first day and sold on the second day | |
if k == 2 and n == 1: | |
return x[1] - x[0] | |
# recursive | |
return max(prevS[n], prevB[n] + x[k-1]) | |
import copy | |
import numpy as np | |
import sys | |
N = int(sys.argv[1]) | |
x = [] | |
for idx in range(N): | |
x.append(np.random.randint(0, N)) | |
print(x) | |
B = [] | |
S = [] | |
for k in range(1, N+1): | |
prevB = copy.copy(B) | |
prevS = copy.copy(S) | |
B = [] | |
S = [] | |
for n in range(int((k+1)/2+1)): | |
B.append(maxCashAfterBuy(x, k, n, prevB, prevS)) | |
S.append(maxCashAfterSell(x, k, n, prevB, prevS)) | |
B.append(-float('inf')) | |
print(S) |
$ python dp2.py 10
[4, 1, 4, 6, 2, 7, 4, 9, 7, 4]
[0, 8, 12, 15, 12, 6]
That is, given 10-day stock price [4, 1, 4, 6, 2, 7, 4, 9, 7, 4], the most profit you can make by trading up to 5 buy-sell trades using aforementioned rule is 0, 8, 12, 15, 12, and 6. Now, to find the answer to the original question, we simply take the maximum of the array. Given 2 maximum buy-sell trades, the most profit one can make is 12.
By the way, the problem become trivial if maximum allowed trade is N/2, in which case one simply buys at local minimum and sells at local maximum. Hence, this algorithm shines when n is less than N/2.
No comments:
Post a Comment