UPDATE: Daniel Ziegler found a way to improve my query time from \(O(log(n)^3)\) to \(O(log(n)^2)\). I’ll write it up eventually. If you need it for some reason in the meantime feel free to email me.
Suppose I want to maintain a set of people where each has an age and a wealth. I want to be able to quickly insert people, delete people, and answer queries of the form “find the \(k\)th richest person whose age is between \(x\) and \(y\)”.
Here is a summary of solutions to different variants on this question:
Variation | Solution details | Space | Update time | Query time |
---|---|---|---|---|
No insertion or deletion, one-sided interval | Persistent binary search trees | $$ n \cdot \log(n)$$ | N/A | $$ \log(n)$$ |
No insertion or deletion, two-sided interval | Persistent binary search trees | $$ n \cdot \log(n)$$ | N/A | $$ \log(n)$$ |
Insertion, deletion, two-sided interval | My answer, presented here | $$ n \cdot \log(n)$$ | $$ \log^2(n)$$ | $$ \log^3(n)$$ |
I have been somewhat interested in this question for about a year, and I’ve asked about related questions a few times. But no-one’s ever managed to give me a complete answer.
The other day, I ran across this StackOverflow answer, which presents an answer which I modified to make a full solution, which I’ll present here.
My solution allows insertion and deletion in \(O(\log^2(n))\), and allows the query in \(O(\log^3(n))\).
Solution
Store an order statistic tree ordered on age. At every node, store a pointer to an auxiliary order statistic tree, of all of that node’s descendants ordered on income.
At every node, this requires an extra amount of memory which is linear in the number of descendants of that node. So this means that the tree will take \(O(n\cdot \log(n))\) memory overall.
Query
You might want to go read that second StackOverflow answer again, because this algorithm is similar to that one, and it’s easier to understand the algorithm on arrays.
The query is similar to how queries across ranges of an augmented BST usually work: we start out by finding the set of nodes whose descendants contain the whole subsection of the tree that you care about. There will be \(\log(n)\) of these nodes, and finding all of them takes \(\log(n)\) time.
We end up with \(\log(n)\) OSTs of maximum size \(O(n)\), and we want to find the \(k\)th smallest item in their disjoint union.
As discussed here, we can solve that query in \(O\left(\log(n)^3\right)\).
Alternatively, if your \(k\) is small, you can directly traverse the trees to find the correct answer, which takes \(k \log(n) + n\) time.
Updates
When I insert a new value into my set, I need to update the auxiliary OSTs of all of the ancestor nodes of my new node.
Usually, it’s easy to argue that maintaining auxiliary data in your OST is fast, because usually your auxiliary data is something like “the sum of your descendants” or something which is \(O(1)\). In this data structure, the efficiency argument is somewhat more complicated.
Inserting a single item into an OST takes \(O(\log(n))\) time. But making the OST from scratch takes \(O(n)\). This is concerning because it means that tree rotations are potentially extremely expensive. If I had to do tree rotations all the way up from my new node to the root of the tree every single insertion, then insertion would take linear time.
Luckily, we can decide that our OST is balanced using the red-black tree rules. Insertion in a red-black tree only involves amortized \(O(1)\) rotations. (See here for an explanation of this.)
The node at the lowest level will have to totally regenerate its auxiliary OST every insertion, of course. Its parent will have to do a tree rotation which requires it to totally regenerate its auxiliary OST every second insertion. Its grandparent will need to do that \(\frac14\) of the time. And then \(\frac18\) and so on.
At height \(h\) in the tree, defining the leaves to be \(h=0\), the amortized cost of insertion is going to be \(O(\log(2^h)) = O(h)\) for insertion plus \(O\left(\frac{2^h}{2^h}\right) = O(1)\) for totally recreating the OST after a rebalance, for a total cost of \(O(h)\).
The total time required for updating all the auxiliary OSTs after an insert is therefore:
$^$O\left(\sum_{h=0}^{log(n)} h \right)= O\left(\log(n)^2\right)$^$
Updating or deleting a node also takes \(O\left(\log(n)^2\right)\), for the same reason.
Variations
limited \(k\)
If \(k\) is always going to fixed below a particular limit \(l\)–say, you know ahead of time that you’re never going to need to know farther back than the 50th richest person between two ages–each node in your main OST can store a smaller auxiliary tree with only \(l\) elements in it.
This reduces memory requirements to \(O(n \cdot l)\).
Queries: Using the algorithm for OSTs in this table, the cost is now \(\log(\log(n) \cdot l) \cdot \log(n) \cdot \log(l)\), which looks like \(\log(\log(n)) \cdot \log(n) \cdot \log(l)\) as \(n\) grows.
Updates: Every ancestor needs to do \(\log(l)\) work now, instead of \(\log(n)\), but you still have \(\log(n)\) ancestors. So update takes overall \(\log(n)\log(l)\) time.