Swift version: 5.6

A monad is any data type that can be mapped over using `map()`

and flat mapped over using `flatMap()`

, as long it abides by three laws. Arrays, sets, optionals, and more are all monads.

You don’t need to understand (or even be aware of) the three monad laws in order to use them, but if you’re curious I’ll try to explain. The three monad laws are best demonstrated using code, because honestly it’s a bit heavy when you’re just learning.

The first law is left identity, and means that if you have: 1) a value, e.g. the number 5; 2) a monad that contains that value, e.g. an array containing the number 5; and 3) a function that accepts the same type of value (5) and sends back the same type of monad (an array); then calling `flatMap()`

on the array should be equal to applying the function directly to the value.

In code:

```
// if you have a value, in this case 5
let myNumber = 5
// and you have a monad containing that value, in this case an array containing 5
let myMonad = [myNumber]
// and you have a function that accepts a number and returns the same type of monad as we had before (an array)
let doubleNumbers = { (value: Int) in return [value * 2] }
// then calling flatMap on the array…
let result1 = myMonad.flatMap(doubleNumbers)
// should be equal to applying the function directly to the value
let result2 = doubleNumbers(myNumber)
// so, this should print "true" in a playground
result1 == result2
```

The second law is right identity, and means that if you have: 1) a value, e.g. the number 5; 2) a monad that contains that value, e.g. an array containing the number 5; and 3) a function that accepts the same type of value (5) and sends back the same kind of monad (an array) without transforming the value; then calling `flatMap()`

with that function on your monad should leave it unchanged.

In code:

```
// if you have a value, in this case 5
let value = 5
// and you have a monad containing that value, in this case an array containing 5
let array = [5]
// and you have a function that accepts a number and returns the same type of monad as we had before (an array) without transforming the value
let wrapInArray = { (value: Int) in return [value] }
// then calling flatMap() with that function on your monad should leave it unchanged
let flatMapped = array.flatMap(wrapInArray)
// this should print "true" in a playground
array == flatMapped
```

The third law is associativity, and means that if you have 1) a value, e.g. the number 5; 2) a monad that contains that value, e.g. an array containing the number 5; and 3) two functions that can be run on that monad as a chain; then it shouldn’t matter how those functions are nested.

```
// if you have a value, in this case 5
let anotherNumber = 5
// and you have a monad containing that value, in this case an array containing 5
let anotherArray = [myNumber]
// and you have two functions that can be run on that monad as a chain, in this case one that multiplies by 5 and one by 10
let multiplyBy5 = { [$0 * 5] }
let multiplyBy10 = { [$0 * 10] }
// then it shouldn’t matter how those functions are nested
let chained = anotherArray.flatMap(multiplyBy5).flatMap(multiplyBy10)
let nested = anotherArray.flatMap { multiplyBy5($0).flatMap(multiplyBy10) }
// this should print "true" in a playground
chained == nested
```

Again, you don’t need to understand these laws in order to use monads, so don’t be too worried if you understood only part of the code above.

**SPONSORED** Superwall lets you build & test paywalls without shipping updates. Run experiments, offer sales, segment users, update locked features and more at the click of button. Best part? It's FREE for up to 250 conversions / mo and the Superwall team builds out 100% custom paywalls – free of charge.

Sponsor Hacking with Swift and reach the world's largest Swift community!

Available from iOS 8.0 – learn more in my book **Pro Swift**

- What is MVVM?
- How to force a crash using fatalError()
- How to store NSCoding data using Codable
- How to create Quick Look debug previews for your custom types
- How to add warnings and errors to your code using #warning and #error

This is part of the Swift Knowledge Base, a free, searchable collection of solutions for common iOS questions.

Link copied to your pasteboard.