From Haskell to Racket
Table of contents
Basic Values
Let’s start by looking at something you know: Haskell. In Haskell, expressions can include literals for numbers, strings, booleans. Here we are using the Haskell’s GHCi which provides a read-eval-print-loop (REPL) to type in examples and evaluate their results:
-- Haskell
> 8
8
> "haskell"
"haskell"
> True
True
> False
False
Note, evaluating values in GHCi gives the value back. Despite Haskell being a typed language, GHCi does not print the types of the expressions by default. We can use :set +t
to require GHCi to print the type of the expression it evaluates.
-- Haskell
> :set +t -- tell GHCi to automatically show types
> 8
8
it :: Num a => a
> "haskell"
"haskell"
it :: [Char]
> True
True
it :: Bool
> False
False
it :: Bool
GHCi now prints the type of the expression it evaluated as it :: Type
. Here it
refers to the last evaluated expression. Note how 8
is not printed as an Integer
but as Num a => a
. It means 8
has a type a
where a
is an instance of the Num
typeclass. The Num
typeclass describes common properties of various kinds of number, such as for example. The native machine integer type Int
, arbitrary-sized integers Integer
, and even floating point type Double
are instances of Num
.
The Racket REPL also operates similarly:
;; Racket
> 8
8
> "racket"
"racket"
> #t
#t
> #f
#f
Racket only prints the value as it is an untyped language. The notation for booleans is slightly different, but both languages agree on numbers, strings, and booleans. The languages are essentially the same so far.
Basic Operations
Haskell uses an infix notation for writing operations.
-- Haskell
> 1 + 2
3
it :: Num a => a
The order of operations follows the usual mathematical precendence rules (which you must memorize), or you can use parentheses to indicate grouping:
-- Haskell
> 1 + (2 * 2)
5
it :: Num a => a
> (1 + 2) * 2
6
it :: Num a => a
Extraneous parenthesis are fine:
-- Haskell
> (((1))) + ((2 * 2))
5
it :: Num a => a
Compared to many languages you may know, including Haskell, Racket employs a uniform, minimalistic concrete syntax based on the concept of parenthesized, prefix notation.
In this notation, parentheses play a much more central role. They are not optional and they signal the form of the expression.
Languages, like people, descend from their ancestors and inherit some of their properties. In the case of notation, Racket inherits the Lisp (and Scheme) notation for programs. It takes a bit of getting used to, but once aclimated, the notation should feel lightweight and consistent; there is verry little to memorize when it comes to syntax.
So in Racket, we would write:
;; Racket
> (+ 1 (* 2 2))
5
> (* (+ 1 2) 2)
6
Note that there are no precendence rules for addition and multiplication: the form of the expression makes it unambiguous.
Parenthesis indicate function applications, so adding extraneous parens means something different than in Haskell:
;; Racket
> (1)
application: not a procedure;
expected a procedure that can be applied to arguments
given: 1
Functions
Haskell also has a notation for writing functions:
-- Haskell
> \x y -> x + y
<interactive>:18:1: error:
* No instance for (Show (Integer -> Integer -> Integer))
arising from a use of 'print'
(maybe you haven't applied a function to enough arguments?)
* In a stmt of an interactive GHCi command: print it
The REPL prints the values of the expression it evaluated. Once functions are evaluated, their source cannot be printed. However, we can still ask GHCi to print the type of the expression by preceeding it with a :t
.
-- Haskell
> :t \x y -> x + y
(\x y -> x + y) :: Num a => a -> a -> a
This make an anonymous function that consumes two numbers and produces their sum.
To apply it, we can write it justapoxed with arguments:
-- Haskell
> (\x y -> x + y) 3 4
7
it :: Num a => a
Note that in Haskell, every function is a function of exactly one argument. Therefore \x y -> x + y
is actuallty shorthand for \x -> \y -> x + y
.
Applying such a function to fewer than 2 arguments will do a partial function application, which will produce a function that take the remaining arguments:
-- Haskell
> :t (\x y -> x + y) 3
(\x y -> x + y) 3 :: Num a => a -> a
To encode functions that must always be given two arguments, a tuple can be used:
-- Haskell
> :t \(x, y) -> x + y
\(x, y) -> x + y :: Num a => (a, a) -> a
To apply such a function, it must be given a pair of integers:
-- Haskell
> (\(x, y) -> x + y)(3, 4)
7
it :: Num a => a
The use of (x, y) here in the function parameters is actually a pattern. This can be understood as shorthand for:
-- Haskell
> :t \p -> case p of (x, y) -> x + y
\p -> case p of (x, y) -> x + y :: Num a => (a, a) -> a
So even this function is actually taking a single argument (which must be a pair of numbers).
Racket has a similar notation for writing functions:
;; Racket
> (λ (x) (λ (y) (+ x y)))
#<procedure>
You can also write this without the fancy λ
by spelling it lambda
:
;; Racket
> (lambda (x) (lambda (y) (+ x y)))
#<procedure>
(In DrRacket, to insert a λ
press Cmd + )
To apply it, it must be written in parens, juxtaposed with arguments:
;; Racket
> (((λ (x) (λ (y) (+ x y))) 3) 4)
7
Functions in Racket do not always consume a single argument. They can consume 0, 1, or more arguments.
;; Racket
> (λ (x y) (+ x y))
#<procedure>
This is not a shorthand for the function above it; rather it is a function that expects two arguments:
;; Racket
> ((λ (x y) (+ x y)) 3 4)
7
Applying a function to the wrong number of arguments will result in an error (and not perform partial function application):
;; Racket
> ((λ (x y) (+ x y)) 3)
arity mismatch;
the expected number of arguments does not match the given number
expected: 2
given: 1
Definitions
In Haskell, variables can be defined with let
:
> let x = 3
x :: Num a => a
> let y = 4
y :: Num a => a
> x + y
7
it :: Num a => a
> :{
let fact = \n -> case n of
0 -> 1
n -> n * (fact (n - 1))
:}
fact :: (Eq a, Num a) => a -> a
> fact 5
120
it :: (Eq a, Num a) => a
The :{
and :}
marks the start and end of a multi-line definition in the REPL.
In Racket, variables are defined with the define
form:
> (define x 3)
> (define y 4)
> (+ x y)
7
> (define fact
(λ (n)
(match n
[0 1]
[n (* n (fact (- n 1)))])))
> (fact 5)
120
In Haskell, function definitions can be written as:
let fact n = case n of
0 -> 1
n -> n * (fact (n - 1))
This is just a shorthand for the definition written above in terms of \
.
Similarly in Racket, function definitions can be written as:
> (define (fact n)
(match n
[0 1]
[n (* n (fact (- n 1)))]))
which is shorthand for the definition above using λ.
Notice both Haskell and Racket have pattern matching forms, which are quite useful for writing function in terms of a number of “cases.” More on this in a minute.
Lists
Haskell has a built-in list datatype. The empty list is written []
and :
is an operation for “consing” an element on to a list. So to build a list with three integer elements, 1, 2, and 3, you’d write:
> 1 : 2 : 3 : []
[1,2,3]
it :: Num a => [a]
Racket has a built-in list datatype. The empty list is written '()
and cons
is an operation for consing an element on to a list. To build the same list, you’d write:
> (cons 1 (cons 2 (cons 3 '())))
'(1 2 3)
The notation (list 1 2 3)
is shorthand for the above.
There is a slight difference here. For one, Haskell lists must be homogeneous. You can have a list of strings or a list of numbers, but you can’t have a list of strings and numbers.
> ["a", 3]
<interactive>:45:7: error:
* No instance for (Num [Char]) arising from the literal '3'
* In the expression: 3
In the expression: ["a", 3]
In an equation for 'it': it = ["a", 3]
In Racket, there is no such restriction:
> (list "a" 3)
'("a" 3)
Also, in Racket, cons
plays the role of both tupling (making pairs) and making lists (making a pair of an element and another list).
So in Haskell, you could make a pair ("a", 3)
. In Racket, you’d write (cons "a" 3)
. Note this is a pair and not a proper list. In Haskell, tuples and lists are disjoint things. In Racket, lists and tuples (pairs) are made out of the same stuff.
This can be confusing the first time you encounter it, so let’s go over it a bit more.
In Racket (or any Lisp), cons
plays the role of both the pair constructor and the list constructor. Non-empty lists are a subset of pairs: they are pairs whose second component is a list (either the empty list or another pair whose second component is a list, etc.).
You can make pairs out of any kind of element and you can make lists out of any kind of elements. We can precisely define these sets as:
;; type ListofAny =
;; | '()
;; | (cons Any ListofAny)
;; type PairofAny =
;; | (cons Any Any)
Or, to give more useful parameterized definitions:
;; type (Listof A) =
;; | '()
;; | (cons A (Listof A))
;; type (Pairof A B) =
;; | (cons A B)
The functions first
and rest
operate on non-empty lists, producing the first element of the list and the tail of the list, respectively.
> (first (cons 3 (cons 4 '())))
3
> (rest (cons 3 (cons 4 '())))
'(4)
These function will produce errors if given something that is a pair but not a list:
> (first (cons 3 4))
first: contract violation
expected: (and/c list? (not/c empty?))
given: '(3 . 4)
> (rest (cons 3 4))
rest: contract violation
expected: (and/c list? (not/c empty?))
given: '(3 . 4)
On the other hand, the functions car
and cdr
access the left and right components of a pair (the names are admittedly awful and an artifact of Lisp history):
> (car (cons 3 4))
3
> (cdr (cons 3 4))
4
When given pairs that are also lists, they behave just like first
and rest
:
> (car (cons 3 (cons 4 '())))
3
> (cdr (cons 3 (cons 4 '())))
'(4)
Pattern Matching
Haskell has a very nice pattern matching for letting you express case analysis and decomposition in a concise way.
Each pattern maching expression has a sub-expression that produce a value to be matched against and a number of clauses. Each clause has a pattern and an expression. The pattern potentially consists of data constructors, variables, and literals. If the value matches the first pattern, meaning the value and the template match up on constructors and literals, then the variables are bound to the correspond parts of the value, and the right-hand side expression is evaluated. If the value doesn’t match, the next pattern is tried, and so on. It’s an error if none of the patterns match.
So for example, we can write a function that recognize even digits as:
> :{
let evenDigit n = case n of 0 -> True
2 -> True
4 -> True
6 -> True
8 -> True
_ -> False
:}
evenDigit :: (Eq a, Num a) => a -> Bool
The patterns here, save the last one, are just integer literals. If n
is the same as any of these integers, the value True
is produced. The last case uses a “wildcard,” which matches anything and produces False
.
Here’s an example that matches a tuple, binding each part of the tuple to a name and then using those names to construct a different tuple:
> let swap p = case p of (x, y) -> (y, x)
swap :: (b, a) -> (a, b)
Here the pattern uses a data constructor (the tuple constructor). It matches any value that is made with the same constructor.
Here is a recursive function for computing the sum of a list of numbers, defined with pattern matching:
> :{
let addNums xs = case xs of [] -> 0
x:xs -> x + (addNums xs)
:}
addNums :: Num p => [p] -> p
> addNums [4,5,6]
15
it :: Num p => p
We can do the same in Racket:
> (define (even-digit n)
(match n
[0 #t]
[2 #t]
[4 #t]
[6 #t]
[8 #t]
[_ #f]))
> (define (swap p)
(match p
[(cons x y) (cons y x)]))
> (define (sum xs)
(match xs
['() 0]
[(cons x xs)
(+ x (sum xs))]))
> (sum (list 4 5 6))
15
Datatypes
Haskell has the ability to declare new datatypes. For example, we can define type for binary trees of numbers:
data BinaryTree = Leaf
| Node Integer BinaryTree BinaryTree
This declares a new type, named BinaryTree
. There are two variants of the BinaryTree
type, each with their own constructor: Leaf
and Node
. The Leaf
constructor takes no arguments, so just writing Leaf
creates a (empty) binary tree:
> :t Leaf
Leaf :: BinaryTree
The Node
constructor takes three arguments: an integer and two binary trees. Applying the constructor to a tuple of three things, makes a (non-empty) binary tree:
> :t Node 3 Leaf Leaf
Node 3 Leaf Leaf :: BinaryTree
Binary trees are an example of a recursive datatype, since one of the variants contains binary trees. This means we can build up arbitrarily large binary trees by nesting nodes within nodes:
> :t Node 3 (Node 4 Leaf Leaf) (Node 7 Leaf Leaf)
Node 3 (Node 4 Leaf Leaf) (Node 7 Leaf Leaf) :: BinaryTree
Pattern matching is used to do case analysis and deconstruct values. So for example, a function that determines if a binary tree is empty can be written as:
> :{
let btEmpty bt = case bt of Leaf -> True
Node _ _ _ -> False
:}
btEmpty :: BinaryTree -> Bool
> btEmpty Leaf
True
it :: Bool
> btEmpty (Node 4 Leaf Leaf)
False
it :: Bool
The patterns use the constructor names to discriminate on which constructor was used for a given binary tree. The use of the wildcard here is just saying it doesn’t matter what’s inside a node; if you’re a node, you’re not empty.
Recursive functions work similarly, but use variables inside patterns to bind names to the binary trees contained inside a node:
> :{
let btHeight bt = case bt of Leaf -> 0
Node _ l r -> 1 + (max (btHeight l) (btHeight r))
:}
btHeight :: (Num p, Ord p) => BinaryTree -> p
> btHeight Leaf
0
it :: (Num p, Ord p) => p
> btHeight (Node 4 (Node 2 Leaf Leaf) Leaf)
2
it :: (Num p, Ord p) => p
We do something very similar in Racket using structures. A structure type is like a (single) variant of a data type in Haskell: it’s a way of combining several things into one new kind of value.
> (struct leaf ())
> (struct node (i left right))
This declares two new kinds of values: leaf structures and node structures. For each, we get a constructor, which is a function named after the structure type. The leaf
constructor takes no arguments. The node constructor takes 3 arguments.
> (leaf)
(leaf)
> (node 5 (leaf) (leaf))
(node 5 (leaf) (leaf))
> (node 3 (node 2 (leaf) (leaf)) (leaf))
(node 3 (node 2 (leaf) (leaf)) (leaf))
There is no type system in Racket, but we can conceptually still define what we mean in a comment. Just like in Haskell, we can use pattern matching to discriminate and deconstruct:
;; type BinaryTree = (leaf | (node Integer BinaryTree BinaryTree))
> (define (bt-empty? bt)
(match bt
[(leaf) #t]
[(node _ _ _) #f]))
> (bt-empty? (leaf))
#t
> (bt-empty? (node 5 (leaf) (leaf)))
#f
> (define (bt-height bt)
(match bt
[(leaf) 0]
[(node _ left right)
(+ 1 (max (bt-height left)
(bt-height right)))]))
> (bt-height (leaf))
0
> (bt-height (node 4 (node 2 (leaf) (leaf)) (leaf)))
2
Symbols
One of the built-in datatypes we will use often in Racket is that of a symbol. A symbol is just an atomic peice of data. A symbol is written using the quote
notation (quote symbol-name)
, which is abbreviated 'symbol-name
. What’s allowable as a symbol name follows the same rules as what’s allowable as a Racket identifier.
Symbols don’t have a whole lot of operations. The main thing you do with symbols is tell them apart from eachother:
> (equal? 'fred 'fred)
#t
> (equal? 'fred 'wilma)
#f
It is possible to convert between symbols and strings:
> (symbol->string 'fred)
"fred"
> (string->symbol "fred")
'fred
There’s also a convient function that produces a symbol that is guaranteed to have not been used so far each time you call it:
> (gensym)
'g2803
> (gensym)
'g2804
> (gensym)
'g2805
They can be used to define “enum” like datatypes:
; type Flintstone = 'fred | 'wilma | 'pebbles
You can use pattern matching to match symbols:
> (define (flintstone? x)
(match x
['fred #t]
['wilma #t]
['pebbles #t]
[_ #f]))
> (flintstone? 'fred)
#t
> (flintstone? 'barney)
#f
There’s really not a precise analog to symbols in Haskell.
Quote, quasiquote, and unquote
One of the distinguishing features of languages in the Lisp family (such as Scheme and Racket) is the quote
operator and its closely related cousins quasiquote
, unquote
, and unquote-splicing
.
Let’s start with quote
.
The “tick” character 'd
is used as a shorthand for (quote d)
.
You’ve already seen it show up with symbols: 'x
is the symbol x
. It also shows up in the notation for the empty list: '()
.
But you can also write quote around non-empty lists like '(x y z)
. This makes a list of symbols. It is equivalent to saying (list 'x 'y 'z)
.
In fact, you can nest lists within the quoted list: '((x) y (q r))
. This is equivalent to (list (list 'x) 'y (list 'q 'r))
.
Here’s another: '(() (()) ((())))
. This is equivalent to
(list '() (list '()) (list (list '())))
So, anything you can write with quoted lists, you can write without quoted lists by pushing the quote inward until reaching a symbol or an empty set of parenthesis.
You can also put strings, booleans, and numbers inside of a quote
. As you push the quote inward, it simply disappears when reaching a string, boolean or number. So '5
is just 5
. Likewise '#t
is #t
and '"Fred"
is "Fred"
.
You can also write pairs with quote
, which uses the .
notation for separating the left and right part of the pair. For example, '(1 . 2)
is equivalent to (cons 1 2)
. If you write something like '(1 2 3 . 4)
, what you are in effect saying is (cons 1 (cons 2 (cons 3 4)))
, an improper list that ends in 4
.
In essence, quote
is a shorthand for conveniently constructing data and is a very concise notation for writing down ad-hoc data. It serves much the same purpose as formats like JSON and XML, except there’s even less noise.
To summarize, with quote
, you can construct
- strings
- booleans
- numbers
- symbols
- and… pairs (or lists) of those things (including this one)
The kind of things you can construct with the quote
form are often called s-expressions, short for symbolic expressions.
We can give a type definition for s-expressions:
; type S-Expr =
; | String
; | Boolean
; | Number
; | Symbol
; | (Listof S-Expr)
The reason for this name is because anything you can write down as an expression, you can write down inside a quote
to obtain a data representation of that expression. You can render an expression as a symbolic representation of itself.
For example, (+ 1 2)
is an expression. When run, it applies the function bound to the variable +
to the arguments 1
and 2
and produces 3
. On the other hand: '(+ 1 2)
constructs a peice of data, namely, a list of three elements. The first element is the symbol +
, the second element is 2
, the third element is 3
.
We will be using (subsets of) s-expressions extensively as our data representation of AST and IR expressions, so it’s important to gain a level of fluency with them now.
Once you understand quote
, moving on to quasiquote
, unquote
, and unquote-splicing
are pretty straight-forward.
Let’s start with quasiquote
. The “backtick” character `d
is used as a shorthand for (quasiquote d)
and the “comma” character ,e
is shorthand for (unquote e)
. The (quasiquote d)
form means the same thing as (quote d)
, with the exception that if (unquote e)
appears anywhere inside d
, then the expression e
is evaluated and it’s value will be used in place of (unquote e)
.
This gives us the ability to “escape” out of a quoted peice of data and go back to expression mode.
If we think of quasiquote
like quote
in terms of “pushing in” then the rules are exactly the same except that when a quasiquote
is pushed up next to an unquote
, the two “cancel out.” So `,e
is just e
.
For example, `(+ 1 ,(+ 1 1))
is equivalent to (list '+ 1 (+ 1 1))
, which is equivalent to (list '+ 1 2)
.
So if quote
signals us to stop interpreting things as expressions, but instead as data, quasiquote
signals us to stop interpreting things as expression, but instead as data.. unless we encounter a unquote, in which case you go back to interpreting things as expressions.
The last remaining peice is unquote-splicing
, which is abbreviated with “comma-at”: ,@e
means (unquote-splicing e)
. The unquote-splicing
form is like unquote
in that if it occurs within a quasiquote
, it means we switch back in to expression mode. The difference is the expression must produce a list (or pair) and the elements of that list (or pair) are spliced in to the outer data.
So for example, `(+ 1 ,@(map add1 '(2 3)))
is equivalent to (cons '+ (cons 1 (map add1 (list 2 3))))
, which is equivalent to (list '+ 1 3 4)
, or '(+ 1 3 4)
.
If the expression inside the unquote-splicing
produces something other than a pair, an error is signalled.
Poetry of s-expressions
The use of structures lets us program in a style very similar to idiomatic Haskell programming. For each variant data type, we can define a structure type for each variant and use pattern matching to process such values.
However, we are going to frequently employ a different idiom for programming with recursive variants which doesn’t rely on structures, but rather uses symbols in place of constructors and lists in place of fields.
Let’s revisit the binary tree example, using this style.
Notice that leaf
structure is a kind of atomic data. It doesn’t contain anything and its only real purpose is to be distinguishable from node
structures. On the other hand a node
structure needs to be distinguishable from leaf
s, but also contain 3 peices of data within it.
We can formulate definition of binary trees using only symbols and lists as:
;; type BinaryTree = 'leaf | (list 'node Integer BinaryTree BinaryTree)
So the following are binary trees:
> 'leaf
'leaf
> (list 'node 3 'leaf 'leaf)
'(node 3 leaf leaf)
> (list 'node 3
(list 'node 7 'leaf 'leaf)
(list 'node 9 'leaf 'leaf))
'(node 3 (node 7 leaf leaf) (node 9 leaf leaf))
This formulation has the added benefit that we write binary trees as s-expressions:
> 'leaf
'leaf
> '(node 3 leaf leaf)
'(node 3 leaf leaf)
> '(node 3
(node 7 leaf leaf)
(node 9 leaf leaf))
'(node 3 (node 7 leaf leaf) (node 9 leaf leaf))
We re-write our functions to match this new datatype definition:
> (define (bt-empty? bt)
(match bt
['leaf #t]
[(cons 'node _) #f]))
> (bt-empty? 'leaf)
#t
> (bt-empty? '(node 3
(node 7 leaf leaf)
(node 9 leaf leaf)))
#f
> (define (bt-height bt)
(match bt
['leaf 0]
[(list 'node _ left right)
(+ 1 (max (bt-height left)
(bt-height right)))]))
> (bt-height 'leaf)
0
> (bt-height '(node 3
(node 7 leaf leaf)
(node 9 leaf leaf)))
2
We even can use quasiquote notation in patterns to write more concise definitions:
> (define (bt-empty? bt)
(match bt
[`leaf #t]
[`(node . ,_) #f]))
> (bt-empty? 'leaf)
#t
> (bt-empty? '(node 3
(node 7 leaf leaf)
(node 9 leaf leaf)))
#f
> (define (bt-height bt)
(match bt
[`leaf 0]
[`(node ,_ ,left ,right)
(+ 1 (max (bt-height left)
(bt-height right)))]))
> (bt-height 'leaf)
0
> (bt-height '(node 3
(node 7 leaf leaf)
(node 9 leaf leaf)))
2
Moreover, we can embrace quasiquotation at the type-level and write:
; type BinaryTree = `leaf | `(node ,Integer ,BinaryTree ,BinaryTree)
Testing, modules, submodules
We will take testing seriously in this class. Primarily this will take the form of unit tests, for which we will use the rackunit
library. To use the library, you must require
it.
Here is a simple example:
> (require rackunit)
> (check-equal? (add1 4) 5)
> (check-equal? (* 2 3) 7)
--------------------
FAILURE
name: check-equal?
location: eval:76:0
actual: 6
expected: 7
--------------------
The check-equal?
function takes two arguments (and an optional third for a message to display should the test fail) and checks that the first argument produces something that is equal?
to the expected outcome given as the second argument.
There are many other forms of checks and utilities for building up larger test suites, but check-equal?
will get us a long way.
As a matter of coding style, we will place tests nearby the function they are testing and locate them within their own module. Let’s talk about modules for a minute.
In Racket, a module is the basic unit of code organization. Every file is a module whose name is derived from the filename, but you can also write modules without saving them in a file. For example:
> (module bt racket
(provide bt-height)
(define (bt-height bt)
(match bt
[`leaf 0]
[`(node ,_ ,left ,right)
(+ 1 (max (bt-height left)
(bt-height right)))])))
This declares a module named bt. It provides a single value named bt-height.
We can require the module from the REPL to gain access to the modules provided values:
> (require 'bt)
> (bt-height 'leaf)
0
We could have also used the #lang racket shorthand for (module bt racket ...)
and saved this in a file called bt.rkt
. To import from a file in the current directory, you’d write (require "bt.rkt")
. But this doesn’t work well in REPL.
For the most part we will organize our programs into single module files using the #lang racket shorthand. But we will place tests within a “sub”-module, i.e. a module nested inside of the module that contains the code it tests. We will use a special form called module+
which declares a submodule that has access to the enclosing module. Moreover, repeated uses of module+
will add content to the submodule. By convention, we will name the testing submodule test
.
So here’s a second version of the bt
module with unit tests included (and more code). Note the use of all-defined-out
to provide everything:
> (module bt2 racket
; provides everything defined in module
(provide (all-defined-out))
(module+ test
(require rackunit))
(define (bt-empty? bt)
(match bt
['leaf #t]
[(cons 'node _) #f]))
(module+ test
(check-equal? (bt-empty? 'leaf) #t)
(check-equal? (bt-empty? '(node 3
(node 7 leaf leaf)
(node 9 leaf leaf)))
#f))
(define (bt-height bt)
(match bt
[`leaf 0]
[`(node ,_ ,left ,right)
(+ 1 (max (bt-height left)
(bt-height right)))]))
(module+ test
(check-equal? (bt-height 'leaf) 0)
; intentionally wrong test:
(check-equal? (bt-height '(node 3 leaf leaf)) 2)))
Requiring this module with make bt-height, but it will not run the tests:
> (require 'bt2)
Running the tests only happens when the test submodule is required:
> (require (submod 'bt2 test))
--------------------
FAILURE
name: check-equal?
location: eval:80:0
actual: 1
expected: 2
--------------------
Putting it all together, we can write the following code and save it in a file called bt.rkt. (You can click the tiny clipboard icon on top-right to copy it.)
#lang racket
(provide (all-defined-out))
(module+ test
(require rackunit))
;; type Bt =
;; | `leaf
;; | `(node ,Integer ,Bt ,Bt)
;; Bt -> Boolean
;; Is the binary tree empty?
(define (bt-empty? bt)
(match bt
['leaf #t]
[(cons 'node _) #f]))
(module+ test
(check-equal? (bt-empty? 'leaf) #t)
(check-equal? (bt-empty? '(node 3
(node 7 leaf leaf)
(node 9 leaf leaf)))
#f))
;; Bt -> Natural
;; Compute the height of a binary tree
(define (bt-height bt)
(match bt
[`leaf 0]
[`(node ,_ ,left ,right)
(+ 1 (max (bt-height left)
(bt-height right)))]))
(module+ test
(check-equal? (bt-height 'leaf) 0)
(check-equal? (bt-height '(node 3 leaf leaf)) 1)
(check-equal? (bt-height '(node 2 leaf (node 1 leaf leaf)))
2))
This code follows a coding style that we will use in this course:
- it’s organized in a module,
- data type definitions occur at the top of the file,
- it uses a test submodule to group unit tests,
- tests occur immediately after the functions they test,
- functions are annotated with type signatures and short purpose statements, and
- indentation follows standard conventions (which DrRacket can apply for you).
From the command line, you can run a module’s tests using the Racket command line testing tool raco test
:
$ raco test bt.rkt
raco test: (submod "bt.rkt" test)
5 tests passed
Or simply give a directory name and test everything within that directory:
$ raco test .
raco test: (submod "./bt.rkt" test)
5 tests passed
These notes are adapted from CMSC430 at UMD.