Back to index import Prelude hiding (not, head, tail)
So far, we have not seen ways of building functions outside of composing already defined operator. In this note, we introduce the constructs of haskell that allow for branching. There will be the standard if/then/else construct, and an idiomatic way of handling boolean conditions with what are called guards. But an arguably more important notion will be pattern matching, which allows to define to define a function by “inspecting the shape of the argument”.
In Haskell, we can use an if then else construct which generally has the following shape.
if boolean then
expr₁
else
expr₂
where boolean
is an expression of type Bool
and expr₁
and expr₂
have the same type, say A.
This results in an overall expression of type A. Let us give a couple
simple examples
integersFromToZero :: Int -> [Int]
= if n >= 0 then
integersFromToZero n 0 .. n-1]
[else
+1 .. 0]
[n
fizzBuzz :: (Integral a, Show a) => a -> String
= let divisibleBy n m = mod n m == 0 in
fizzBuzz x if 3 `divisibleBy` x then
if 5 `divisibleBy` x then
"fizzbuzz"
else
"fizz"
else if 5 `divisibleBy` x then
"buzz"
else
show x
-- the following is not an example of good style, but might be instructive as
-- an example of a valid definition.
greetings :: Bool -> Bool -> String
= (if a then "Hi " else "Hello") ++
greetings a b " " ++
if b then "friend" else "stranger"
The following is not valid as both branches do not have the same type:
= if x == 0 then
someIncorrectCode x "78"
else 78
Note also that something that is not allowed is having an if/then, but no else branches, as in
= if x /= 0 then
someIncorrectCode2 x x
If you do this, you will have a syntax error, possibly pointing to the next non-empty line of your program.
If you have an if/then/else at the toplevel of your expression, there is another idiomatic piece of syntax you might want to employ called guards. The syntax for a definition using a guard is the following
fnName args | condition₁ = body₁
| condition₂ = body₂
...
that should be read as “in case conditioni is true, then fnName args is defined to be bodyi”. We can reformulate the first two examples using guards as follows:
integersFromToZeroG :: Int -> [Int]
| n >= 0 = [0 .. n-1]
integersFromToZeroG n | otherwise = [n+1 .. 0]
divisibleBy :: Integral a => a -> a -> Bool
`divisibleBy` m = mod n m == 0
n
fizzBuzzG :: Int -> String
| 3 `divisibleBy` x = if 5 `divisibleBy` x then
fizzBuzzG x "fizzbuzz"
else
"fizz"
| 5 `divisibleBy` x = "buzz"
| otherwise = show x
Note that the otherwise
constant is simply an alias for
True
.
>>> otherwise
True
Another thing to note is that the first available line is taken as the definition, as is probably obvious from how the “otherwise” cases are laid out.
Unlike with if/then/else, there is no obligation of coverage of all cases with guards. Here is an example
takeVariant :: [Int] -> [Int]
| head xs >= 0 = take (head xs) (tail xs)
takeVariant xs | null xs = []
>>> takeVariant [2 .. 10]
[3,4]
Finally, there is another, more primitive idiom to define functions by cases. The idea is that, in a function declaration
... | condition = body fnName arg₁ arg₂
each of arg₁ can be replaced by a pattern that describes the shape of an argument. We will later see how and why this is the most primitive way of doing a conditional in Haskell, but we need to see how to do so in a few particular cases:
For our first example, we can give a couple of examples where the pattern is a constant.
not :: Bool -> Bool
not True = False
not False = True
incrIfNZero :: Int -> Int
0 = 0
incrIfNZero = n + 1 incrIfNZero n
Note that the order of the cases matters, much like with guards. The following two examples illustrate that
>>> incrIfNZero 0
0
>>> incrIfNZero 1
2
Note that in some cases, the branching allowed by the pattern-matching means you do not need to record some function arguments to know what is the output. In that case, one may replace a variable by an underscore to indicate “there is an argument, but I don’t even care to give it a name”. Here is an example of the logical function corresponding to implication.
impl :: Bool -> Bool -> Bool
False _ = True
impl = b impl _ b
Now, pattern-matching is especially powerful with structured type constructors like tupling and lists. Let us start with tupling. Besides constants, one can go a step further with tupling: we can start describing the shape of the tuple and give variable names to the components. Here is an example.
sumTriple :: Num a => (a, a, a) -> a
= x + y + z sumTriple (x, y, z)
Here we gave the names x y and z to the components of a single argument. This is true for triples of arbitrary size. It is also possible to nest patterns and actually analyze nested tuples easily.
assoc :: ((a, b), c) -> (a, (b, c))
= (x, (y, z)) assoc ((x, y), z)
In patterns, one can only use each variables once; if you want to check equality of parts of arguments like in the following example, you have to use conditionals or guards:
addDistincts :: (Eq a, Num a) => (a, a) -> a
| x == y = x
addDistincts (x, y) | otherwise = x + y
The reason we can do such pattern-matching is because, from the point of view of the haskell runtime, every fully evaluated thing of a tuple type must have shape (-, -) and (- , -) is a special basic construcor of the tuple types.
There is something similar going on with lists. In that case, there
are two primitive list constructors that can be pattern-matched against:
the empty list []
and the cons operator
(:)
:t [] [] :: [a]
:t (:) (:) :: a -> [a] -> [a]
Here is an example
takeVariant2 :: [Int] -> [Int]
= [21]
takeVariant2 [] : xs) = take x xs takeVariant2 (x
Note that the parentheses around the list patterns are necessary
(otherwise the compiler would try to parse x as an argument and would
run into a syntax error when seeing the infix operator :
in
the middle fo the left-hand side)
In fact the basic function head and tail are defined using pattern-matching in the prelude (note that definitions with pattern-matching need not to cover all arguments and still be legal).
head :: [a] -> a
head (x : _) = x
tail :: [a] -> [a]
tail (_ : xs) = xs
Much like with tuples, list patterns can be nested. Furthermore (and this is a special syntax for lists), one may use a syntax with brackets to match against list of fixed length.
addListOfSizeFiveOnly :: Num a => [a] -> a
= x + y + z + t + u
addListOfSizeFiveOnly [x,y,z,t,u] = 0
addListOfSizeFiveOnly _
firstTwoEqual :: Eq a => [a] -> Bool
: y : _) = x == y
firstTwoEqual (x = False
firstTwoEqual _
someFunction :: [([a], [a])] -> [([a],[a])]
: xs) = ([x], y : z) : xs
someFunction (([x,y], z) : ys) : xs) = ([t], ys) : xs
someFunction (([], t = [] someFunction _
One can also do pattern-matching, as well as guards, in definitions in where clauses and let-ins
someOtherFunction :: (Num a, Ord a) => [a] -> a
= let f (x : _) | x > 0 = x
someOtherFunction xs = 0 in
f []
g (f xs)where g 0 = 2
= 2 - x g x
Finally, there is also a way to do a patter matching in the middle of a function using a construct called case. You can look up the syntax online, but be careful if you use it as it is sensitive to indentation, much like where clauses, so it’s easy to make mistakes that lead to mysterious syntax errors.
someDefinition :: [Int] -> [Char]
= "abc" ++ case takeVariant xs of
someDefinition xs -> "cba"
[] : xs -> show x
x
someOtherDefinition :: [Int] -> Int
= case takeVariant2 (xs ++ [2 .. 5]) of
someOtherDefinition xs : z : ys | y > z -> y
y | otherwise -> z
-> 2 []