Disjoint Types
Objectives
When you are done with this chapter, you will….
- explain what a disjoint type is.
- describe the components of a disjoint type:
- type names, constructors, and parameters.
- use disjoint types to declare linked lists, options, and binary trees.
- write functions that traverse and manipulate them.
- explain how memory is allocated when data is immutable.
What are Disjoint Types?
A disjoint type is a type that can take one or more non-overlapping forms. For example, one representation of a list is either an empty list or a pair where the first element is a datum and the second is a reference to another list. Another example is a binary tree, where you have nodes containing data and references to children, or else a leaf node that has neither. In our class we will use disjoint types to represent expression trees.
In contrast, a complex number is not a disjoint type. It has a real part and an imaginary part, but these always come together. This applies also to data structures like pairs, arrays, or strings. Each of these may have components, but there is fundamentally only one representation for them.
Syntax
The keyword data
starts a disjoint type declaration. You then supply a name (capitalized) followed by zero or more type variables (lower case).
After that you declare one or more constructors (capitalized), each with zero or more type names to be used as positional parameters.
That will make more sense if you see some examples.
Example 1 — Integer Lists
For our first example, we will make a linked list similar to the built-in one, but it will only accept integers.
data IList = ICons Integer IList
| INil
deriving (Eq,Show)
- The name of the type is
IList
. There are no type parameters for this type. - The first constructor is
ICons
and takes two parameters: anInteger
and anotherIList
. - The second constructor is
INil
and takes no parameters. - The
deriving (Show,Eq)
tells Haskell to defineshow
and equality operations so we can print these out and compare them.
You can think of IList
as being something like a class in object oriented languages, and ICons
and INil
being two kinds of constructors for it.
Calling either one of these will get you an IList
, and in fact these are functions.
Prelude> :t INil
INil :: IList
Prelude> :t ICons
ICons :: Integer -> IList -> IList
Let’s create a few lists:
Prelude> i1 = INil
Prelude> i2 = ICons 10 (ICons 20 INil)
Prelude> i3 = ICons 44 i2
Prelude> i3
ICons 44 (ICons 10 (ICons 20 INil))
Variable i1
just contains an INil
. Variable i2
is a list containing 10 and 20. Variable i3
contains 44, 10, and 20.
Memory Considerations
It is helpful if you understand how memory is allocated. The rule is simple: whenever you see the name of a constructor in an
expression context (i.e., not as a function pattern) it allocates memory. If you see a variable then it is a reference, sharing
the memory. So there are five memory allocations in the above code. Variables i2
and i3
share the list containing 10 and 20.
Example 2 — List of Anything
Example 3 — Options
Example 4 — Binary Trees
Example 5 — Something Crazy
Here’s an example:
data Foo a b = Bar Integer a
| Baz b
| Quux
deriving (Show,Eq)
In this example, Foo
is the name of the type, and it takes two type variables a
and b
. The three constructors are Bar
, Baz
, and Quux
.
As you see, it’s possible for a constructor to take no parameters at all.
To instantiate this, you can use the REPL
.
Prelude> x = Bar 10 True
Prelude> y = Baz "Hi"
Prelude> z = Quux
Prelude> x
Bar 10 True
Prelude> y
Baz "Hi"
Prelude> z
Quux
Prelude> :t x
x :: Foo Bool b
Prelude> :t y
y :: Foo a [Char]
Prelude> :t z
z :: Foo a b
Notice that Bar
takes two parameters, and the type of the second parameter determines the type of type parameter a
. The second type b
is was unspecified by
x
so it is left there. This is similar to what you saw with the empty list having type [a]
earlier.
You can use a type declaration to force the unconstrained part if necessary:
Prelude> a = Bar 10 True :: Foo Bool String
Prelude> a
Bar 10 True
Prelude> :t a
a :: Foo Bool String
It