Exercises from PFDS Section 2.2

Posted on Dec 4, 2012

Section 2.2’s exercises define some optimizations to the section’s unbalanced tree set implementation.

Exercise 2.2

The implementation of member that Okasaki gives for binary search trees in section 2.2 performs 2d comparisons for a tree of depth d in the worst case, when searching for a number that is the farthest to the right on the tree, and when the right path in the tree is of depth d itself. This is because every call of member checks whether x < y and, if not, whether y < x. This strategy allows immediate detection of the case in which the value being searched for is contained in the current node, permitting short-circuit cases like where the searched-for value is in the root node, at the cost of requiring double the comparisons when traversing rightward.

Exercise 2.2 proposes a strategy in which the second comparison is delayed until the bottom of the tree. This removes the 2d worst-case number of comparisons, at the cost of raising the minimum number of comparisons to the number of nodes in the shortest path from the root to a leaf. The implementation is relatively straightforward:

class UnbalancedTreeSet[T] private (tree: BinaryTree[T]) (implicit val ordering: Ordering[T]) extends Set[T] {
// ...
def member(x: T) = UnbalancedTreeSet.member(x, tree, None)
// ...
}
object UnbalancedTreeSet {
// ...
private def member[T](x: T, t: BinaryTree[T], c: Option[T])(implicit ordering: Ordering[T]): Boolean = (t, c) match {
case (BinaryTreeLeaf, None) => false
case (BinaryTreeLeaf, Some(d)) => ordering.equiv(x, d)
case (BinaryTreeNode(a, y, b), _) =>
if (ordering.lt(x, y)) member(x, a, c)
else member(x, b, Some(y))
}
// ...
}

We introduce a new parameter to member, c: Option[T], which is None when there is not yet a candidate for equality, and Some(d) when d is a candidate for equality. Then, when we reach a leaf node, we check to see whether our candidate is equal to the value we’re searching for.

Exercise 2.3

Exercise 2.3 proposes an optimization to insert. The implementation given by Okasaki performs wasteful path copying when the value being inserted is already in the tree. If the value being inserted is present in a particular node, the only thing that’s done is to short-circuit the operation and return the node as it exists, which doesn’t reverse any of the path copying performed while traversing the tree down to the node.

Throwing an exception in the case where the element being inserted already exists avoids the wasteful path copying. Okasaki requires that the function establish only one exception handler per insert, not per function call, so we catch the exception in the helper insert method of the UnbalancedTreeSet class.

class ElementAlreadyExistsException extends RuntimeException
class UnbalancedTreeSet[T] private (tree: BinaryTree[T]) (implicit val ordering: Ordering[T]) extends Set[T] {
// ...
def insert(x: T) = try {
newSet(UnbalancedTreeSet.insert(x, tree))
} catch {
case _:ElementAlreadyExistsException => this
}
// ...
}
object UnbalancedTreeSet {
// ...
private def insert[T](x: T, t: BinaryTree[T])(implicit ordering: Ordering[T]): BinaryTree[T] = t match {
case BinaryTreeLeaf => BinaryTreeNode(BinaryTreeLeaf, x, BinaryTreeLeaf)
case BinaryTreeNode(a, y, b) =>
if (ordering.lt(x, y)) BinaryTreeNode(insert(x, a), y, b)
else if (ordering.lt(y, x)) BinaryTreeNode(a, y, insert(x, b))
else throw new ElementAlreadyExistsException
}
// ...
}

Exercise 2.4

Exercise 2.4 is to integrate the improvements to member from exercise 2.2 into the improved version of insert from 2.3. This is straightforward: we follow the pattern established by 2.2, threading an equality candidate through the recursive calls, and checking for equality only at the end. Now insert performs no unnecessary path copying if inserting a pre-existing element, and performs at most d + 1 comparisons along the way.

class ElementAlreadyExistsException extends RuntimeException
class UnbalancedTreeSet[T] private (tree: BinaryTree[T]) (implicit val ordering: Ordering[T]) extends Set[T] {
// ...
def insert(x: T) = try {
newSet(UnbalancedTreeSet.insert(x, tree, None))
} catch {
case _:ElementAlreadyExistsException => this
}
// ...
}
object UnbalancedTreeSet {
// ...
private def insert[T](x: T, t: BinaryTree[T], c: Option[T])(implicit ordering: Ordering[T]): BinaryTree[T] =
(t, c) match {
case (BinaryTreeLeaf, None) => BinaryTreeNode(BinaryTreeLeaf, x, BinaryTreeLeaf)
case (BinaryTreeLeaf, Some(d)) =>
if (ordering.equiv(x, d)) throw new ElementAlreadyExistsException
else BinaryTreeNode(BinaryTreeLeaf, x, BinaryTreeLeaf)
case (BinaryTreeNode(a, y, b), _) =>
if (ordering.lt(x, y)) BinaryTreeNode(insert(x, a, c), y, b)
else BinaryTreeNode(a, y, insert(x, b, Some(y)))
}
// ...
}

Exercise 2.5

Exercise 2.5 is about generating balanced binary trees that share as much as possible. The exercise assumes that all of the values in the trees are the same, which doesn’t make much sense for the set abstraction, but ensures that all subtrees of the same size are identical.

For the first part, we implement a function complete that creates a complete tree of d levels. As a binary tree, it should have 2^d - 1 values inside. The implementation is a straightforward recursive function. The base case, a binary tree of 0 levels, is just a leaf node. The recursive case creates an internal node whose children are both the result of recursing with one less level.

For the second part, we implement a function balanced that creates a tree with d values that is as balanced as possible: every internal node’s subtrees differ in size by at most 1.

I put both of these methods directly on BinaryTree’s companion object. I think this is idiomatic, but it’s not really clear to me: there are so many different ways to do things in Scala.

object BinaryTree {
def complete[T](x: T, d: Int): BinaryTree[T] = d match {
case 0 => BinaryTreeLeaf
case _ => {
val subtree = complete(x, d - 1)
BinaryTreeNode(subtree, x, subtree)
}
}
def balanced[T](x: T, d: Int)(implicit ordering: Ordering[T]): BinaryTree[T] = d match {
case 0 => BinaryTreeLeaf
case 1 => BinaryTreeNode(BinaryTreeLeaf, x, BinaryTreeLeaf)
case _ => {
val half = d / 2
val halftree = create(x, half)
if (half * 2 == d) BinaryTreeNode(halftree, x, create(x, half - 1))
else BinaryTreeNode(halftree, x, halftree)
}
}
}

I wrote a few tests for the tree generation functions in a normal assertion-based style before remembering the existence of ScalaCheck and its obvious applicability for this kind of thing. I’ve been itching to use a QuickCheck-style testing framework for a while. If you haven’t seen it before, you should definitely check it out. It allows you to write tests in a declarative style, without specifying individual test cases. The framework will generate random test cases for you, either by using built-in random generators for simple cases like arbitrary strings or integers, or by using a user-defined random object generator.

In this case, since I wanted my test cases to vary the number of levels or nodes in a balanced tree, I was able to just use the built-in random integer generator. Writing tests in this way is concise and fun. Here’s what they ended up looking like:

class BinaryTreeTest extends Spec with ShouldMatchers with Checkers {
describe("A complete tree") {
it("should have 2^n - 1 nodes") {
check(forAll(choose(0, 25)){n: Int =>
BinaryTree.complete(0, n).size == scala.math.pow(2, n) - 1 })
}
}
describe("A balanced tree") {
it("should have n nodes") {
check(forAll(choose(0, 100)){n: Int =>
BinaryTree.balanced(0, n).size == n})
}
}
}

Exercise 2.6

This exercise asks the reader to modify UnbalancedTreeSet so that it represents a map instead of a set. This is very straightforward, if tedious: we must add another data member to BinaryTreeNode that represents the value of an entry in the map, change member to lookup and have it return the value data member instead of true, and throw an exception instead of false, and finally change insert to bind, give it an extra value argument, and set the BinaryTreeNode’s value field to that argument. I didn’t implement this.

As usual, the new code’s on GitHub.