CH 14
CH 14
CH 14
Exercise 14.1-1
Exercise 14.1-2
Exercise 14.1-3
Exercise 14.1-5
Exercise 14.1-6
First perform the usual BST insertion procedure on z, the node to be in-
serted. Then add 1 to the rank of every node on the path from the root to z
1
Algorithm 1 O
S-SELECT’(x,i)
r = x.left.size
while i 6= r do
if i > r then
x = x.right
i = i-r
else
x=x.left
end if
r = x.left.size
end while
return x
Algorithm 2 RECURSIVE-OS-KEY-RANK(T,k)
1: if T.root.key == k then
2: Return T.root.lef t.size + 1
3: else if T.root.key > k then
4: Return RECU RSIV E − OS − KEY − RAN K(T.root.lef t, k)
5: else
6: Return RECU RSIV E − OS − KEY − RAN K(T.root.right, k) +
T.root.lef t.size + 1
7: end if
2
such that z is in the left subtree of that node. Since the added node is a leaf,
it will have no subtrees so its rank will always be 1. When a left rotation is
performed on x, its rank within its subtree will remain the same. The rank
of x.right will be increased by the rank of x, plus one. If we perform a right
rotation on a node y, its rank will decrement by y.lef t.rank + 1. The rank of
y.lef t will remain unchanged. For deletion of z, decrement the rank of every
node on the path from z to the root such that z is in the left subtree of that
node. For any rotations, use the same rules as before.
Exercise 14.1-7
See the algorithm INV-COUNT(L). It does assume that all the elements of
the list are distinct. To adapt it to the not neccesarily distinct case, each time
that the we do the search, we should be selecting the element with that key that
comes first in an inorder traversal.
Algorithm 3 INV-COUNT(L)
Construct an order statistic tree T for all the elements in L
t = −|L|
for i from 0 to |L| − 1 do
t = t + OS − RAN K(T, SEARCH(T, L[i]))
remove the node corresponding to L[i] from T
end for
return t
Exercise 14.1-8
Choose a point on the circle and assign it key value 1. The rank of any point
will be its position when read in the sequence starting with point 1, and reading
clockwise around the circle. Next, we label each point with a key. Reading
clockwise around the circle, if a point’s chordal companion has a key, assign it
the same key. Otherwise, assign it the next lowest unused integer key value.
Then use the algorithm given in the solution to problem 2-4 d to count the
number of inversions (based on key value), with the caveat that an inversion
from key i to key j (with i > j)doesn’t count if the rank of the companion of i
is smaller than the rank of j.
Exercise 14.2-1
Along all the nodes in an inorder traversal, add a prev and succ pointer. If
we keep the head to be the first node, and, make it circularly linked, then this
clearly allows for the four operations to work in constant time. We still need
to be sure to maintain this linked list structure throughout all the tree modifi-
cations. Suppose we insert a node into the BST to be a left child, then we can
3
insert it into this doubly linked list immediately before it’s parent, which can
be done in constant time. Similarly, if it is a right child, then we would insert it
immediately after its parent. Deletion of the element is just the usual deletion
in a linked list.
Exercise 14.2-2
Since the black height of a node depends only on the black height and color
of its children, Theorem 14.1 implies that we can maintain the attribute with-
out affecting the asymptotic performance of the other red-black tree operations.
The same is not true for maintaining the depths of nodes. If we delete the root
of a tree we could potentially have to update the depths of O(n) nodes, making
the DELETE operation asymptotically slower than before.
Exercise 14.2-3
After performing the rotate operation, starting at the deeper of the two nodes
that were moved by the rotate, say x, set x.f = x.lef t.f ⊗x.a⊗x.right.f . Then,
do the same thing for the higher up node in the rotation. For size, instead set
x.size = x.lef t.size + x.right.size + 1 and then do the same for the higher up
node after the rotation.
Exercise 14.2-4
Algorithm 4 RB-ENUMERATE(x,a,b)
if a ≤ x.key ≤ b then
print x
end if
if a ≤ x.key and x.lef t 6= N IL then
RB − EN U M ERAT E(x.lef t, a, b)
end if
if x.key ≤ b and x.right 6= N IL then
RB − EN U M ERAT E(x.right, a, b)
end if
Return
Exercise 14.3-1
after rearranging the nodes, starting with the lower of the two nodes moved,
4
set it’s max attribute to be the maximum of its right endpoint and the the two
max attributes of its children. Do the same for the higher up of the two moved
nodes.
Exercise 14.3-2
Exercise 14.3-3
Consider the usual interval search given, but, instead of breaking out of the
loop asa soon as we have an overlap, we just keep track of the most recently
seen overlap, and keep going in the loop until we hit T.nil. We then return the
most recently seen overlap. We have that this is the overlapping interval with
minimum left end point because the search always goes to the left it contains an
overlapping interval, and the left children are the ones with smaller left endpoint.
Exercise 14.3-4
Algorithm 5 INTERVAL(T,i)
if T.root overlaps i then
Print T.root
end if
if T.root.lef t 6= T.nil and T.root.lef t.max ≥ i.low then
INTERVAL(T.root.left,i)
end if
if T.root.right 6= T.nil and T.root.right.max ≥ i.low and T.root.right.key ≤
i.high then
INTERVAL(T.root.right,i)
end if
The algorithm examines each node at most twice and performs constant
time checks so the runtime cannot exceed O(n). If a recursive call is made on
a branch of the tree, then that branch must contain an overlapping interval, so
the runtime also cannot exceed O(k lg n) since the height is at most n and there
are k intervals in the output list.
Exercise 14.3-5
We could modify the interval tree insertion procedure to, once you find
a place to put the given interval, then we perform an insertion procedure
based on the right hand endpoints. Then, to perform INTERVAL-SEARCH-
EXACTLY(T,i) first perform a search for the left hand endpoint, then, perform
a search for the right hand endpoint based on the BST rooted at this node for
5
the right hand endpoint, stopping the search if we ever come across a element
with a different left hand endpoint.
Exercise 14.3-6
Store the elements in a red-black tree, where the key value is the value of
each number itself. The auxiliary attribute stored at a node x will be the min
gap between elements in the subtree rooted at x, the maximum value contained
in subtree rooted at x, and the minimum value contained in the subtree rooted
at x. The min gap at a leaf will be ∞. Since we can determine the attributes of
a node x using only the information about the key at x, and the attributes in
x.lef t and x.right, Theorem 14.1 implies that we can maintain the values in all
nodes of the tree during insertion and deletion without asymptotically affecting
their O(lg n) performance. For MIN-GAP, just check the min gap at the root,
in constant time.
Exercise 14.3-7
Let L be the set of left coordinates of rectangles. Let R be the set of right
coordinates of rectangles. Sort both of these sets in O(n lg(n)) time. Then, we
will have a pointer to L and a pointer to R. If the pointer to L is smaller, call
interval search on T for the up-down interval corresponding to this left hand
side. If it contains something that intersects the up-down bounds of this rect-
angle, there is an intersection, so stop. Otherwise add this interval to T and
increment the pointer to L. If R is the smaller one, remove the up-down interval
that that right hand side corresponds to and increment the pointer to R. Since
all the interval tree operations used run in time O(lg(n)) and we only call them
at most 3n times, we have that the runtime is O(n lg(n)).
Problem 14-1
b. We will present a simple solution to this problem that runs in time O(n lg(n))
which doesn’t augment a red-black tree even though that is what is suggested
by the hint. Consider a list of elements so that each element has a integer
x.pos and a field that says whether it is a left endpoint or a right endpoint
x.dir = L or R. Then, sort this list on the pos attribute of each element.
Then, run through the list with a running total of how many intervals you
are currently in, subtracting one for each right endpoint and adding one for
each left endpoint. Also keep track of the running max of these values, and
6
the endpoint that has that value. Then, this point that attains the running
max is what should be returned.
Problem 14-2
b. Begin by creating an order statistic tree for the given elements, where rank
starts at 0 and ends at n − 1, the rank of a node is its position in the original
order minus 1, and we store each element’s value in an attribute x.value.
Print the element with rank m(n) − 1, then delete it from the tree. Then
proceed as follows: If you’ve just printed the k th value which had rank r,
delete that node from the tree, and the (k + 1)st value to be printed will have
rank r − 1 + m(n) mod (n − k). Since deletion and lookup take O(lg n) and
there are n nodes, the runtime is O(n lg n).