Stock price processing problem
Question
You are given a stream of records about a particular stock. Each record contains a timestamp and
the corresponding price of the stock at that timestamp.
Unfortunately due to the volatile nature of the stock market, the records do not come in order.
Even worse, some records may be incorrect.
Another record with the same timestamp may appear later in the stream
correcting the price of the previous wrong record.
Design an algorithm that:
Updates the price of the stock at a particular timestamp,
correcting the price from any previous records at the timestamp.
Finds the latest price of the stock based on the current records.
The latest price is the price at the latest timestamp recorded.
Finds the maximum price the stock has been based on the current records.
Finds the minimum price the stock has been based on the current records.
Implement the StockPrice class:
StockPrice() Initializes the object with no price records.
void update(int timestamp, int price) Updates the price of the stock at the given timestamp.
int current() Returns the latest price of the stock.
int maximum() Returns the maximum price of the stock.
int minimum() Returns the minimum price of the stock.
Example 1:
Input
["StockPrice", "update", "update", "current", "maximum", "update", "maximum", "update", "minimum"]
[[], [1, 10], [2, 5], [], [], [1, 3], [], [4, 2], []]
Output
[null, null, null, 5, 10, null, 5, null, 2]
Explanation
StockPrice stockPrice = new StockPrice();
stockPrice.update(1, 10); // Timestamps are [1] with corresponding prices [10].
stockPrice.update(2, 5); // Timestamps are [1,2] with corresponding prices [10,5].
stockPrice.current(); // return 5, the latest timestamp is 2 with the price being 5.
stockPrice.maximum(); // return 10, the maximum price is 10 at timestamp 1.
stockPrice.update(1, 3); // The previous timestamp 1 had the wrong price, so it is updated to 3.
// Timestamps are [1,2] with corresponding prices [3,5].
stockPrice.maximum(); // return 5, the maximum price is 5 after the correction.
stockPrice.update(4, 2); // Timestamps are [1,2,4] with corresponding prices [3,5,2].
stockPrice.minimum(); // return 2, the minimum price is 2 at timestamp 4.
Constraints:
1 <= timestamp, price <= 109
At most 105 calls will be made in total to update, current, maximum, and minimum.
current, maximum, and minimum will be called only after the update has been called at least once.
Solution
Initial approach (Failed on reserved ordered stocks by price)
Even worse, some records may be incorrect. Another record with the same timestamp may appear later in the stream correcting the price of the previous wrong record.
When the next stock price overwrites the existing MAX price or MIN price for the same timestamp to a smaller or larger price, we need to find the next largest one or next smallest one to provide the correct MAX or MIN value.
This means we need to keep the previous prices in order and should be able to access both max and min values quickly.
We also need to remember the price for each timestamp for later correction as well.
Given the requirements, I thought of creating the binary search tree with a counter to remember the occurrence of the prices. The reason for having counters was that there could be multiple identical prices for the same timestamp. When the same price as the existing price is added, we increase the count by 1 and when the old price is corrected to a new price, we will decrease the old price by 1 and increase the counter of the new price by 1. If the old price count becomes zero, we will remove it from the tree. If the new price does not exist in the tree, we will add the price to the tree with the count of 1.
With this solution, the time complexity of CURRENT, MAX, and MIN is O(1), O(logN), (logN) respectively.
The only problem with this solution is that I did not have to rebalance logic to keep the height of the binary search tree to stay optimal. Due to that, my solution was too slow for large stock prices coming in descending order thereby making the height of my tree, N and making both MAX and MIN operations run in O(N)
Here is the solution in Python.
class StockPriceWithBST(object): | |
def __init__(self): | |
self.root = None | |
self.latest = None | |
self.records = {} | |
def update(self, timestamp, price): | |
""" | |
:type timestamp: int | |
:type price: int | |
:rtype: None | |
""" | |
if timestamp not in self.records: | |
self.records[timestamp] = price | |
self.add_to_tree(timestamp, price) | |
else: | |
self.update_node(timestamp, price) | |
self.records[timestamp] = price | |
if self.latest is None: | |
self.latest = timestamp | |
elif self.latest == timestamp or self.latest < timestamp: | |
self.latest = timestamp | |
def current(self): | |
""" | |
:rtype: int | |
""" | |
return self.records[self.latest] | |
def maximum(self): | |
""" | |
:rtype: int | |
""" | |
max = self.find_max() | |
return max.v if max != None else None | |
def minimum(self): | |
""" | |
:rtype: int | |
""" | |
min = self.find_min() | |
return min.v if min != None else None | |
def search(self, v): | |
cur = self.root | |
while(cur != None): | |
if cur.v == v: | |
return cur | |
elif cur.v < v: | |
cur = cur.right | |
else: | |
cur = cur.left | |
return None | |
def update_node(self, t, v): | |
original = self.records[t] | |
found = self.search(original) | |
if len(found.children) == 1: | |
#only record for that value. Remove the node from the tree. | |
self.delete(found) | |
self.add_to_tree(t, v) | |
else: | |
# there are more record in that node. Revmoe only that record. | |
del found.children[t] | |
self.add_to_tree(t, v) | |
return found | |
def add_to_tree(self, t, v): | |
if self.root is None: | |
self.root = Node(t, v) | |
# add stock record to the node | |
self.root.children[t] = v | |
else: | |
prev = None | |
cur = self.root | |
# O(log N) | |
while (cur is not None): | |
if cur.v == v: | |
# add to the existing node. | |
cur.children[t] = v | |
return | |
if cur.v > v: | |
# search left | |
prev = cur | |
cur = cur.left | |
else: | |
# search right | |
prev = cur | |
cur = cur.right | |
# did not find the existing value | |
if prev is not None: | |
node = Node(t, v) | |
if prev.v > v: | |
prev.left = node | |
else: | |
prev.right = node | |
node.parent = prev | |
node.children[t] = v | |
def delete(self, cur): | |
if cur is None: | |
return None | |
if cur.left is None: | |
self.transplant(cur, cur.right) | |
elif cur.right is None: | |
self.transplant(cur, cur.left) | |
else: | |
#has both children | |
candidate = self.successor(cur.right) | |
if cur.right != candidate: | |
self.transplant(candidate, candidate.right) | |
candidate.right = cur.right | |
candidate.right.parent = candidate | |
self.transplant(cur, candidate) | |
#move cur's left to new candidate | |
candidate.left = cur.left | |
candidate.left.parent = candidate | |
return cur | |
def successor(self, parent): | |
cur = parent | |
while (cur.left is not None): | |
cur = cur.left | |
return cur | |
def transplant(self, cur, new): | |
if cur.parent is None: | |
self.root = new | |
elif cur.parent.left == cur: | |
cur.parent.left = new | |
else: | |
cur.parent.right = new | |
if new is not None: | |
new.parent = cur.parent | |
def find_max(self): | |
cur = self.root | |
while cur.right != None: | |
cur = cur.right | |
return cur | |
def find_min(self): | |
cur = self.root | |
while cur.left != None: | |
cur = cur.left | |
return cur | |
# Your StockPrice object will be instantiated and called as such: | |
obj = StockPrice() | |
obj.update(1,10) | |
obj.update(2,5) | |
assert 5 == obj.current() | |
assert 10 == obj.maximum() | |
obj.update(1,3) | |
assert 5 == obj.maximum() | |
obj.update(4,2) | |
assert 2 == obj.minimum() |
A better approach using SortedDict
# solution using sorted map and hashmap. | |
from sortedcontainers import SortedDict | |
class StockPrice(object): | |
def __init__(self): | |
self.prices = SortedDict() | |
self.recoreds = {} | |
self.latest = None | |
def update(self, timestamp, price): | |
""" | |
:type timestamp: int | |
:type price: int | |
:rtype: None | |
""" | |
if self.latest is None or self.latest == timestamp or self.latest < timestamp: | |
self.latest = timestamp | |
if timestamp in self.recoreds: | |
old_price = self.recoreds[timestamp] | |
#update the old price | |
if self.prices[old_price] > 1: | |
self.prices[old_price] -= 1 | |
else: | |
del self.prices[old_price] | |
# add the new price | |
self.prices[int(price)] = 1 if int(price) not in self.prices else self.prices[int(price)] + 1 | |
self.recoreds[timestamp] = int(price) | |
def current(self): | |
""" | |
:rtype: int | |
""" | |
return self.recoreds[self.latest] | |
def maximum(self): | |
""" | |
:rtype: int | |
""" | |
value = self.prices.peekitem(-1)[0] | |
return value | |
def minimum(self): | |
""" | |
:rtype: int | |
""" | |
value = self.prices.peekitem(0)[0] | |
return value | |
# Your StockPrice object will be instantiated and called as such: | |
obj = StockPrice() | |
obj.update(1,10) | |
obj.update(2,5) | |
assert 5 == obj.current() | |
assert 10 == obj.maximum() | |
obj.update(1,3) | |
assert 5 == obj.maximum() | |
obj.update(4,2) | |
assert 2 == obj.minimum() |
Comments
Post a Comment