This course page was updated until March 2022 when I left Durham University. For future updates, please visit the new version of the course pages.
Manipulating data types #
A binary tree #
In lectures we saw (repeatedly) a binary tree with values at nodes and empty leaves. This time, we’re going to make a tree that holds values at nodes and leaves, and then do some manipulation of it.
Our data type is
data Tree a = Leaf a | Node (Tree a) a (Tree a)
deriving (Eq, Show)
The template code/trees-exercise6.hs
also defines some helper functions to
construct trees of various kinds
-- Produce a "balanced" tree with the specified number of nodes
makeBalancedTree :: Int -> Tree Int
-- Take a tree and produce a new tree that is mirror-symmetric
mirrorTree :: Tree a -> Tree a
We’ll first implement some type class instances for our data type.
Exercise
Implement
Functor
andFoldable
instances for theTree
data type.Show that your implementation of
fmap
obeys the two functor laws.Hint
We’ll use these instances to implement functionality transforming trees later, so if you couldn’t figure it out, don’t forget that you can add
{-# LANGUAGE DeriveFunctor #-} {-# LANGUAGE DeriveFoldable #-}
At the top of your file and add
deriving (Functor, Foldable)
to thedata
declaration.
To help construct trees, we’ll build minimal depth trees from lists
with toTree
Exercise
Define
toTree :: [a] -> Tree a
That takes a list and builds a tree of minimal depth.
You may find
splitAt
useful.One thing you should check is that
toList . toTree
is the identity on lists.To check that you got things right, also define
depth :: Tree -> Int
that returns the depth of a tree, defined as the length of its longest branch from the root to any leaf.
You should then find that the depth of a tree constructed with
toTree
grows likelogBase 2
of the length of the input list.
We’ll call binary trees balanced if the number of leaves in the left
and right subtrees of every Node
differs by at most one (with leaves
being trivially balanced).
Exercise
Define a function
balanced :: Tree a -> Bool
that determines if a tree is balanced or not.
Hint
Since your datatype isFoldable
you can compute itslength
.
Question
What complexity does your implementation of
balanced
have? Does it traverse the tree only once (so linear complexity in the tree size), or does it traverse many times?Hint
If you usedlength
, think about how that is implemented.If you traversed more than once, can you think of a way to define
balanced
that only visits each node in the tree at most once?Hint
Think about what information you need to return at each level in the recursion so that you don’t need to go back down later to reconstruct things.
We’ll call binary trees symmetric if we can draw a vertical line through the root node such that right subtree is the mirror image of the left subtree.
Exercise
Write a function
symmetric :: Tree a -> Bool
that checks if a binary tree is structurally symmetric (that is the shape is symmetric, but the values may differ).
hint
You might find it helpful to write
mirror :: Tree a -> Tree a -> Bool
to check if two trees are the mirror image of one another.
A simple calculator #
We will now build a tiny domain-specific language for integer arithmetic expressions, and an evaluator for this language. We begin by only supporting addition:
data Expr = Val Int | Add Expr Expr
deriving (Eq, Show)
This says that an expression is either a Val
(which contains
an Int
) or else an Add
which contains two expressions.
We make the expression representing $(1 + (4 + 5))$ like so:
expr = (Add (Val 1) (Add (Val 4) (Val 5)))
First, we will write some individual recursive functions to inspect these expressions in some way. Then we will look at ways of generalising the idea using higher-order functions.
Exercise
Define
- Evaluation of an expression
eval :: Expr -> Int
- Collecting a list of all
Val
nodescollect :: Expr -> [Val]
- Counting how many
Val
nodes there aresize :: Expr -> Int
In writing these functions, you should see that you produce the same
pattern each time. The Val
node is replaced by some new value
constructed from the integer inside it, and the Add
node is
replaced by some new value constructed from the replacement of both
its internal nodes. We will encapsulate this abstraction in the
function
folde :: (Int -> a) -> (a -> a -> a) -> Expr -> a
That is, folde
eats a function Int -> a
that converts Val
nodes into type a
, a function a -> a -> a
that
converts Add
nodes into type a
by eating both their
converted children, and an Expr
, returning a value of type
a
.
As an example, we can implement eval
using this new function
like so:
eval :: Expr -> Int
eval expr = folde id (+) expr
This says: “replace Val
nodes with their contents, and replace
Add
nodes with the summation of their contents”.
Exercise
Implementfolde
and then the functionscollect
andsize
using it.
Solutions
I’ve added some commented solutions to these exercises. If you have queries about them please ask in the practical sessions or else get in touch.