Skip to content

Latest commit

 

History

History
118 lines (74 loc) · 3.97 KB

IO.md

File metadata and controls

118 lines (74 loc) · 3.97 KB

IO

The IO monad is for isolating effects to maintain referential transparency in your software. Essentially you create a description of your effects of which is performed as the last action in your programme. The IO is lazy and will not be evaluated until the perform (alias run) method is called.

Constructors

IO[A](fn: () => A): IO[A]

The constructor for the IO monad. It is a purely functional wrapper around the supplied effect and enables referential transparency in your software.

IO(() => $("#id").val())
IO.io(() => $("#id").val())
IO.of(() => $("#id").val())
IO.unit(() => $("#id").val())
IO.pure(() => $("#id").val())

Creating an IO from a pimped function

.io()

Wraps a supplied function in an IO. Assumes no arguments will be supplied to the function.

(() => $("#id")).io()

.io1()

Returns a function that will return an IO when one parameter is supplied.

(id => $(id)).io1()

or more simply

$.io1()

Methods

flatMap

Aliases: bind, chain

IO[A].flatMap(fn: A => IO[B]): IO[B]

Perform a monadic bind (flatMap) over the effect. It takes a function that returns an IO. This will happen lazily and will not evaluate the effect.

map

IO[A].map(fn: A => B): IO[B]

Performs a map over the result of the effect. This will happen lazily and will not evaluate the effect.

run

Aliases: perform, performUnsafeIO

Evaluates the effect inside the IO monad. This can only be run once in your programme and at the very end.

...and undocumented

  • join
  • takeLeft
  • takeRight
  • ap

Examples

Say we have a function to read from the DOM and a function to write to the DOM. This example uses jQuery.

const read = id => $(id).text()

const write = (id, value) => $(id).text(value)

On their own both functions would have a side effect because they violate referential transparency. The read function is dependent on an ever changing DOM and thus subsequent calls to it would not produce the same result. The write function obviously mutates the DOM and so it too is not referentially transparent, as each time it is called, an effect occurs.

We can modify this functions so that instead of performing these side-effects they will just return an IO with the yet-to-be-executed function inside it.

const read = id => IO(() => $(id).text())

const write = id => value => IO(() => $(id).text(value))

You can call write(id) until you are blue in the face but all it will do is return an IO with a function inside.

We can now call map and flatMap to chain this two effects together. Say we wanted to read from a div covert all the text to uppercase and then write back to that div.

const toUpper = text => text.toUpperCase()

const changeToUpperIO = read("#myId").map(toUpper).flatMap(write("#myId"))

So what is the type of changeToUpperIO? Well it is the IO type. And that means at this stage, nothing has been executed yet. The DOM has not been read from, the text has not been mapped and the DOM has not been updated. What we have is a referentially transparent description of our programme.

In other pure functional languages such as Haskell we would simply return this type back to the runtime, but in JavaScript we have to manage this ourselves. So now let's run our effect.

changeToUpperIO.run()

Now our DOM should be updated with the text converted to upper case.

It becomes much clearer which functions deal with IO and which functions simply deal with data. read and write return an IO effect but toUpper simply converts a supplied string to upper case. This pattern is what you will often find in your software, having an effect when you start (i.e. reading from a data source, network etc), performing transformations on the results of that effect and finally having an effect at the end (such as writing result to a database, disk, or DOM).