Skip to content

Latest commit

 

History

History
651 lines (485 loc) · 25.9 KB

01-values-and-functions.md

File metadata and controls

651 lines (485 loc) · 25.9 KB

Values and functions (& basic types)

Running examples

If you want to run the examples, I recommend installing stack via the link found in the README and using our ready-made template for small applications: stack new project-name quanterall/basic. This will give you an already set-up project and you can put the example code in the Library.hs file.

If you are so inclined, you can use development containers to get a functioning development environment that I've set up; the prompt should pop up when you open the project. If you don't feel like you want this, you can install the "Simple GHC (Haskell) Integration" (extension ID dramforever.vscode-ghc-simple`) in VSCode for a setup that is known to work. When you save you should be able to get compiler warnings and errors.

Most examples have (and should have) the needed imports in the examples themselves, so if you find one that doesn't, please let me know.

If you want to run the examples interactively you can run stack repl in the root directory of the generated project and it will start an interactive session where you can execute the functions that have been defined, as well as all the standard library. If you want to make changes and see these reflected in the session, you can issue the :r command in it and it will reload the project files.

Values

The bread and butter of a Haskell program is of course values and functions.

Before a value or a function definition comes the type of the definition. This is by convention to a large degree, we don't actually have to specify types for a lot of things depending on whether or not the compiler can figure them out itself. Even so, most projects will be set up to warn you when a top-level definition doesn't have an explicit type for the simple reason that it's part of good documentation to specify the intended type.

A type signature is written with the name of the thing it's for, followed by :: and then the type:

myValue :: Int

The actual definition for it comes immediately after and is written with = to signify that the name on the left is equal to the expression on the right:

myValue :: Int
myValue = 42

The basic types in Haskell:

import Prelude

myAnswerToEverything :: Int -- Integer, limited in size
myAnswerToEverything = 42

myInteger :: Integer -- Integer, can grow as needed
myInteger = 42

myChar :: Char -- A single character
myChar = '$'

myFloat :: Float -- 32-bit floating point value
myFloat = 1337.0

myDouble :: Double -- 64-bit floating point value
myDouble = 1337.0

myBool :: Bool -- `True` or `False`
myBool = False

myString :: String -- Actually just a type alias for a list of characters; `[Char]`
myString = "This is not the best string in the world, it's just a tribute"

-- `String`s also handle "exotic" script
helloThere :: String
helloThere = "Ехо... Генерал Кеноби"

Exercises (Values)

  1. Define a value with the name myOwnValue with the type Double and the value 42.

  2. Define a value with the type String and the value ['h', 'e', 'l', 'l', 'o'], then a value with the type String and the value "hello". Why do both of these work?

  3. Define a value with the type String and the value 'hello'. What happens and why do you think that is?

Functions

Functions are written much like values, but have arrows in their type. The types, starting at the leftmost, represent the arguments/parameters passed to the function, with the final (rightmost) one representing the return value.

import Prelude

--          a      b     result
addInts :: Int -> Int -> Int
addInts a b = a + b

Calling functions, as we can see above, can be done via infix notation when they are operators. We could also call the operator in a prefix position:

import Prelude

--          a      b     result
addInts :: Int -> Int -> Int
addInts a b = (+) a b

The application of a function is done by just writing the function name followed by a space and then spaces between the parameters, making it as lightweight an operation as possible.

import Prelude

--                x    divisor result
isDivisibleBy :: Int -> Int -> Bool
isDivisibleBy x divisor =
  -- When we want to divide our problem into smaller parts we can use `let`.
  -- `rem` here is a function that takes an integer and returns the remainder of dividing it by the
  -- second argument. It is the same as the `%` operator (modulo) from C, JavaScript, and so on.
  let remainderOfDivision = rem x divisor
  -- `let` has to be followed by `in` and then the final expression that is to be executed.
   in remainderOfDivision == 0

rem in the above code snippet is a function that takes an integer-like number and returns the remainder of dividing it by the second argument. Here we check whether or not that division returns 0 to determine if it was divisible by it.

We could also write the division as follows:

import Prelude

--                x    divisor result
isDivisibleBy :: Int -> Int -> Bool
isDivisibleBy x divisor =
  -- Note how we surround the function in backticks (`) to be able to put it in the infix position.
  let remainderOfDivision = x `rem` divisor
   in remainderOfDivision == 0

This can be used to make code read more intuitively and is very common in the case of checking if something is a member of a map, list or the like:

key `Map.member` ourMap -- Is the key defined in the map?
element `List.elem` ourList -- Is the element present in the list?

We can use this with any function, even functions with more than two arguments. It's important to recognize that it's not always a great idea to use this feature and each case should be examined on an individual basis in terms of whether or not it makes the code more or less easy to understand.

Boolean & arithmetic operations

Math Haskell Notes
+ +
- -
* *
/ /
> >
< <
>=
<=
= ==
/=
xⁿ x ^ n x is numeric, n is integral
xⁿ x ** n x & n are both floating point values
x mod n x rem n x and n are integral [0]
x mod n x mod n x and n are integral [0]
Math C Haskell Notes
¬ ! not
&& &&
ǀǀ ǀǀ Two pipe characters

[0]:

-- Note: Use `rem` if you know your parameters are both positive

5 `mod` 3 == 2
5 `rem` 3 == 2

5 `mod` (-3) == -1
5 `rem` (-3) == 2

(-5) `mod` 3 == 1
(-5) `rem` 3 == -2

(-5) `mod` (-3) == -2
(-5) `rem` (-3) == -2

Arithmetic can only be done with numbers of the same type, so what happens when we want to divide an integer number with a floating point number? Let's see:

--              integer  float   result
divideInteger :: Int -> Float -> Float
-- This is the same as `(fromIntegral x) / f` because `/` divides everything on the left by
-- everything on the right.
-- `fromIntegral` is a function that takes anything integer-like and turns it into a `Float`.
divideInteger integer float = fromIntegral integer / float

Likewise we can also take a float and turn it into an integer, though this can obviously lead to a loss of precision in our calculations:

subtractRoundedFloat :: Int -> Float -> Int
subtractRoundedFloat int float = int - round float

If we were to run this we'd see:

Q> subtractRoundedFloat 5 5.5
-1
Q> subtractRoundedFloat 5 5.4
0

Exercises (Functions)

  1. Define a function that returns whether or not an Int is zero.

  2. Define a function that returns whether or not a Float is greater than zero.

  3. Define a function that adds 1/10th of a Double to itself.

  4. Define a function that takes 2 Floats length' & width and returns the area of the rectangle they make up.

  5. Define a function that takes a radius of type Float and returns the area of a circle[0].

  6. Define a function calculateBMI that takes a Float representing weight in kilograms and an Int representing height in centimeters and returns the person's BMI.

Exercise notes (Functions)

  1. pi is available by default.

Asking questions about values

Quite regularly we will have to pose some kind of question about the structure of a value, or whether or not it matches some kind of predicate.

if expressions

if in Haskell is a lot like if in other languages, with the one slight difference to some of them that both the True and False branch need to be present, and both branches need to return the same type of value. This is because if, like most other things in Haskell, is an expression and the result can be bound to a name.

value :: Int -> Int
value x = if x < 10 then x else 10

value' :: Int -> Int
value' x =
  if x < 10
    then x
    else 10

Let's look at a few ways to ask questions about values in the case of "clamping" a number.

Guards

import Prelude

-- | Limits a given integer to be within the range @lowerBound <= value <= upperBound@.
clamp :: Int -> Int -> Int -> Int
clamp lowerBound upperBound value
  | value < lowerBound = lowerBound
  | value > upperBound = upperBound
  | otherwise = value

The above is likely the most natural way of doing this in this very example, because we have several questions to ask about the value and we can do so immediately in what are known as "guards".

Note that we do not have an immediate = after our parameters but instead each | introduces a new question that we pose, a new guard. If the boolean expression that follows the pipe (|) evaluates to True the expression to the right of = is what will be returned.

The word otherwise is an always matching case and we can use this case as an "for all other cases" clause.

Exercises (Asking questions about values)

  1. Define a function that takes two Ints and returns the biggest of the two. Implement it both with function guards as well as if.

  2. Define a function that takes two Ints and returns the smallest of the two. Implement it both with function guards as well as if.

  3. Define a function that subtracts an integer from another, but if the result is less than zero, instead return 0.

  4. Define a function that takes an Int and if it's smaller than zero returns 0, if it's bigger than 255 returns 255.

Exercise notes (Asking questions about values)

$ for function application

Sometimes you will see a dollar sign operator ($) in code. This is actually a utility operator meant for function application:

f :: Int -> String
f x = show (deriveFlorbFactorFromMerkleNumber (castToMerkleNumber x))

f' :: Int -> String
f' x = show $ deriveFlorbFactorFromMerkleNumber $ castToMerkleNumber x

As you can see from the above snippet, using $ allows us to just say "Apply the function on the left to the expression on the right". $ binds tightly to the right, so the parentheses we see above is what will be by default, hence we have no need for parentheses like this when we use $.

Partial application

When you apply a function, you can choose to not pass all the arguments it's expecting. This will result in a function that expects the remaining arguments and that will have the same return value:

import Prelude

addInts :: Int -> Int -> Int
addInts a b = a + b

add42 ::          Int -> Int
add42 = addInts 42

Passing only one arguments to addInts in the example above results in a function that expects one more argument instead of the original two.

It's quite common to use this fact by putting important arguments in the last parameter position of a function in order to let people use this pattern:

import qualified Data.List as List
import Prelude

-- | Adds 42 to every item in a list
add42ToAll :: [Int] -> [Int]
add42ToAll = List.map (\x -> x + 42)

In the above example, List.map takes the list it is working with as the last argument, meaning we can just partially apply it and still get the function we expect; a function that expects a list of integer values that we then add 42 to, returning the resulting list. Since we are not passing the list argument to List.map here we get exactly that function signature.

We can also partially apply our +. The function that we are passing to List.map is expected to be of type Int -> Int, which is what we get when we write (+ 42):

import qualified Data.List as List
import Prelude

-- | Adds 42 to every item in a list
add42ToAll :: [Int] -> [Int]
add42ToAll = List.map (+ 42) -- could also be `(42 +)`

Since operators expect arguments both on the left and right side we can partially apply whichever side we want, so (42 +) is also valid. The resulting behavior obviously depends on the operator, as an operator like - would behave differently depending on which side you are omitting.

Exercises (Partial application)

  1. Define a function that takes a list of Ints and multiplies each with 2. Remember map[0].

  2. Define a function that takes a list of Ints and squares each. Remember map[0].

  3. filter[1] over a list of Int takes a function of type Int -> Bool. Construct a function that checks whether or not the number is 0 without naming it.

  4. Define a function that returns all of the Ints in a list over 0, use partial application for both the check and filter[1].

  5. Define a function that takes all the first numbers of a list of Ints below 10, stopping when it reaches one that does not meet that criteria. takeWhile[2] can be useful for this. Use partial application both for takeWhile[2] and the predicate you pass to it.

  6. Define a function that checks whether or not all Ints in a a list are even. Use notes to figure out how and use partial application for your definition.

Exercise notes (Partial application)

  1. map
  2. filter
  3. takeWhile
  4. all
  5. even

Pipelines using partial application

Partial application is especially useful in pipelines, since we can construct new functions by partially applying other functions, easily creating a function that takes the result of a previous operation and returning a new one, possibly passing it along to yet another function:

import Control.Category ((>>>))
import Data.Function ((&))
import Prelude

-- `/=` is the "not equal" operator in Haskell, analogous to `!=` in many other languages.
-- Note how we're again using an operator with only one argument, and are missing the left-most one,
-- which gives us a function expecting one argument, which is exactly what the predicate we pass to
-- `takeWhile` here expects: `Char -> Bool`

-- Types for this example:
--
-- reverse :: String -> String
-- takeWhile :: (Char -> Bool) -> String -> String
-- length :: String -> Int

dataPartLength :: String -> Int
dataPartLength = length . takeWhile (/= '1') . reverse

dataPartLength' :: String -> Int
dataPartLength' = reverse >>> takeWhile (/= '1') >>> length

dataPartLength'' :: String -> Int
dataPartLength'' string = string & reverse & takeWhile (/= '1') & length

In the above examples we are pipelining functions that operate on the result of a previous function call. The first example does this using the . operator, which represents classic function composition. One thing to note about this is that the application order is read from right to left, so we are applying reverse first, then takeWhile, then length.

reverse takes a String and will reverse it. The result is then passed to takeWhile (/= '1') which will take all initial characters of the string until we find one that is '1'. The result of that is then passed to length which will return the length.

The example using >>> does the same thing, but can be read from left to right. The last example, using the operator & is the same as commonly used pipeline operators like |> from F#, Elm & Elixir, and might be more readable to some. It works by taking whatever value we have on the left side of it and passing it to the function on the right. Like F# and Elm the value is passed as the last argument to the function on the right, as opposed to Elixir where it is passed as the first.

While we aren't changing the meaning of our program based on which way we compose our functions, We should always prefer to read our code left-to-right and up-to-down (because this is the already established reading direction in the rest of the language), meaning we should use >>> and &.

A "Project Euler" example

Let's look at the very first Project Euler as an example of using function composition and partial application to get the answer to a mildly complex question:

If we list all the natural numbers below 10 that are multiples of 3 or 5, we get 3, 5, 6 and 9. The sum of these multiples is 23.

Find the sum of all the multiples of 3 or 5 below 1000.

Let's first look at the given example numbers in action:

import Data.Function ((&))
import Prelude

isDivisibleBy :: Int -> Int -> Bool
isDivisibleBy x divisor =
  -- Note how we surround the function in backticks (`) to be able to put it in the infix position.
  let remainderOfDivision = x `rem` divisor
   in remainderOfDivision == 0

solution :: Int
solution =
  [1,2,3,4,5,6,7,8,9] & filter (\x -> x `isDivisibleBy` 3 || x `isDivisibleBy` 5) & sum -- 23

We can use a shorthand plus takeWhile to make this a bit neater:

import Data.Function ((&))
import Prelude

isDivisibleBy :: Int -> Int -> Bool
isDivisibleBy x divisor =
  -- Note how we surround the function in backticks (`) to be able to put it in the infix position.
  let remainderOfDivision = x `rem` divisor
   in remainderOfDivision == 0

solution :: Int
solution =
  -- `[1..]` means "create an infinite list of increasing numbers starting from 1"
  -- We never have an infinite list in memory, but rather take elements one by one until we reach 10
  [1..] & takeWhile (< 10) & filter (\x -> x `isDivisibleBy` 3 || x `isDivisibleBy` 5) & sum -- 23

If we now set an upper bound as a parameter we can get the solution to the actual problem:

import Data.Function ((&))
import Prelude

isDivisibleBy :: Int -> Int -> Bool
isDivisibleBy x divisor =
  -- Note how we surround the function in backticks (`) to be able to put it in the infix position.
  let remainderOfDivision = x `rem` divisor
   in remainderOfDivision == 0

solution :: Int -> Int
solution upperBound =
  -- You could also write `[1..(upperBound - 1)]` and skip `takeWhile`, but for reasons explained
  -- later on this doesn't make a difference in the execution of this function.
  [1..] & takeWhile (< upperBound) & filter (\x -> x `isDivisibleBy` 3 || x `isDivisibleBy` 5) & sum

-- `solution 1000` will give us the value 233168

If we wanted to support using different divisors we could also do the following:

import Data.Function ((&))
import Prelude

isDivisibleBy :: Int -> Int -> Bool
isDivisibleBy x divisor =
  -- Note how we surround the function in backticks (`) to be able to put it in the infix position.
  let remainderOfDivision = x `rem` divisor
   in remainderOfDivision == 0

solution :: Int -> [Int] -> Int
solution upperBound divisors =
  -- `any` takes a predicate/question and a list of inputs and answers the question:
  -- "Are any of these true?"/"Do any of these return `True`?"
  -- Here we are asking "Is X divisble by any of the passed in divisors?"
  -- We could also write `isDivisibleBy x` only, but it can be useful to use backticks to emphasize
  -- that `x` is the first argument, and to make it read as a question about `x`.
  [1..] & takeWhile (< upperBound) & filter (\x -> any (x `isDivisibleBy`) divisors) & sum

-- `solution 1000 [3, 5]` will give us the value 233168

Exercises (Pipelines using partial application)

  1. Define a function using a pipeline that takes a list of Int, gets all the even numbers in it, multiplies them all by two and returns the sum[1]. Define versions that use:

    • a named argument; with &
    • an unnamed argument; with >>>
  2. Define a function using a pipeline that takes a list of Int, gets all the even numbers in the beginning of the list[2], multiplies them all by two, sums them up and returns whether or not the sum is even. Define versions that use:

    • a named argument; with &
    • an unnamed argument; with >>>
  3. Define a function that takes the last 3 elements of a [Int] and gets the average of them. Use notes as inspiration.

  4. Define a function using a pipeline that takes a list of strings, takes all the strings that begin with "#", then discards all leading "#" or " " from the resulting strings. Take note of what kind of structure you are working with and what we need to do to work with the structures inside.

Exercise notes (Pipelines using partial application)

  1. Remember that & is defined in Data.Function and >>> is defined in Control.Category. These can be imported with import Data.Function ((&)) and import Control.Category ((>>>)).
  2. sum
  3. See examples and previous notes for inspiration and help with this
  4. dropWhile
  5. isPrefixOf
  6. any
  7. elem
  8. take
  9. reverse

A note on functions, their parameter order and partial application

Because we have partial application it's very common to design our APIs in such a way where the "subject" of a function comes last. When we design with partial application in mind we can get functions where the arguments act almost like configuration and we create specialized versions of these on-demand just by partially applying them.

If we take our clamp function from before as an example:

-- Limits a value to be within the range `lowerBound <= value <= upperBound`
clamp :: Int -> Int -> Int -> Int
clamp lowerBound upperBound value
  | value < lowerBound = lowerBound
  | value > upperBound = upperBound
  | otherwise = value

clampAllToByteRange :: [Int] -> [Int]
clampAllToByteRange = List.map (clamp 0 255)

clampAllToHundreds :: [Int] -> [Int]
clampAllToHundreds = List.map (clamp (-100) 100)

Because List.map takes the list it is working with as the last argument we can partially apply List.map and because clamp does the same with the value it is clamping we can partially apply that as well. This leads to very easy code-reuse for different concerns.

Don't worry about mis-designing this

If you do happen to mis-design an API with regards to partial application you will definitely feel it. Seeing the issue is trivial and in most cases you'll just swap the order to more effectively make use of partial application. In the absolute worst case you'll simply not use the function with partial application and that's fine too.

It is absolutely worth thinking about this when designing your APIs, but at the end of the day it's not important enough for you to do somersaults in the code base to get it to where it should be.