Variable Declarations and Types
The Beginnings

Objectives

When you are done with this chapter, you will…

  • Declare and initialize variables in a file and in the REPL,
  • Use type declarations to tell the compiler what type a value should be,
  • Explain the difference between a type and a type constraint,
  • Declare values of the following types:
    • Int and Integer
    • Bool
    • Double
    • Char and String
    • Lists
  • and use type inference to avoid declaring types (and explain why you might not want to).

We will look at functions in the next chapter.

Prerequisites

  • You should have installed Haskell on your own computer or have access to it online.
  • You should be able to use the REPL as well as load definitions from a file.

First Declarations

Create a file vars.hs with the following lines in it.

jpn :: Int
jpn = 8675309

answer :: Integer
answer = 42

somePi :: Double
somePi = 3.14159

status :: Bool
status = True

zero :: Char
zero = '0'

This code declares and defines five variables. You’ll notice that there are two lines for each; the first line declares the type, and the second line declares the value. The value declaration must follow immediately after the type declaration. It turns out that in Haskell is really two languages. The “main” language is the language for talking about expressions, and is what people usually mean when they say the word Haskell. The second language is the type language. Most people only use the simpler aspects of it to declare the types of variables and functions, but in fact the type language is also Turing Complete. There is a famous (?) blog post Typing the Technical Interview in which the protagonist solves the 8-queens problem using Haskell’s type language. It looks a little like Prolog.

The first variable is named jpn and has type Int. The type Int represents 32-bit machine-integers.

The second variable is named answer and has type Integer. The type Integer represents the integers you see in Mathematics. They are not bounded by a set number of bits; they are bound only by the amount of memory you have. Some languages call these BigInts, and you may have used them in languages such as Python or Java.

There is also status of Boolean type (note that True and False in this language are capitalized), and zero, a single character. Characters are enclosed in single quotes.

Now you can start up a REPL and interact with the variables. In the code below, the Prelude> and *Main> lines are prompts. The text before it tells you what modules are loaded. We did not declare a module in the file, so Haskell made one by default. You can hit Control-D to exit.

Prelude> :l vars
[1 of 1] Compiling Main             ( vars.hs, interpreted )
Ok, one module loaded.
*Main> status
True
*Main> foo = answer + 1
*Main> foo
43
*Main> :t foo
foo :: Integer

In the above sample run, we create a new variable in the REPL named foo that is equal to answer + 1, or 43. Notice we did not have a type declaration like in the vars.hs file. Haskell has automatic type inference, so it is able to determine the type of an expression by context. It is in fact possible to write an entire program without these type annotations, but unless you are writing a “throwaway” script it is better that you leave the type annotations in, for two reasons: first, the annotations tell the compiler what you intended to do and allow it to detect any errors more quickly and give better error messages; second, the annotation also serves as documentation so you will know what you were thinking when you read the code again in a few years.

Lists and Strings

One data structure you will see all the time in Haskell is the list. One way to create them is to write out the elements between square brackets. It will look like an array from other languages like Python, but in fact they are singly linked lists.

*Main> xx = [foo,3,4]
*Main> :t xx
xx :: [Integer]

Notice that type also is within square brackets; we read this as “a list of integers.” A list in Haskell has to be monomorphic, that is, the contents must all be of the same type. A list like [10,"hi",True] would not be allowed. We will say more about lists in a future chapter, but we need to mention it now because the default representation of strings in Haskell is a list of characters.

Main> greeting = "Hello, world!"
Main> :t greeting
greeting :: [Char]

If you are writing out type declarations, the type String is a synonym for [Char]. Haskell has several other, more efficient representations for strings, but we will not need them in this course.

Type Variables and Constraints

What do you think is the type of an empty list?

Main> xx = []
Main> :t xx
[] :: [a]

Type type a is a type variable. The type [a] means “a list of anything”. You will see type variables a lot, especially when you work with container types or polymorphic functions.

Sometimes you want a type of data that has certain operations available, but you don’t care about the specific type. For example, you might want something that is a number, but you don’t care if it’s an Int, Integer, or Float. Consider this code:

Main> x = 10
Main> :t x
x :: Num p => p
Main> y = [10,30]
Main> :t y
y :: Num a => [a]

The type Num p => p indicates that type p belongs to a type class Num. We refer to these as type classes, or just classes for short, but they are not classes in the object oriented language sense of the word; it is more like a “classification”. The Num p part is called a type constraint, and is similar to interfaces in languages like Java. In this case it says that type p must be in the class Num. For the variable y the constraint is Num a and the type is [a], meaning the list can be considered any type that is numeric.

There are many different type constraints available in Haskell, and you will be writing them yourself in this course.

In the Repl you can use :info Num to see what operations are available for the Num class. It’s a good way to discover functionality you might not have guessed.

Author: Mattox Beckman

Created: 2022-08-08 Mon 14:13