This is part 25 of Categories for Programmers. Previously: F-Algebras. See the Table of Contents.

If we interpret endofunctors as ways of defining expressions, algebras let us evaluate them and monads let us form and manipulate them. By combining algebras with monads we not only gain a lot of functionality but we can also answer a few interesting questions. One such question concerns the relation between monads and adjunctions. As we’ve seen, every adjunction defines a monad (and a comonad). The question is: Can every monad (comonad) be derived from an adjunction? The answer is positive. There is a whole family of adjunctions that generate a given monad. I’ll show you two such adjunction.

Let’s review the definitions. A monad is an endofunctor `m`

equipped with two natural transformations that satisfy some coherence conditions. The components of these transformations at `a`

are:

η_{a}:: a -> m a μ_{a}:: m (m a) -> m a

An algebra for the same endofunctor is a selection of a particular object — the carrier `a`

— together with the morphism:

alg :: m a -> a

The first thing to notice is that the algebra goes in the opposite direction to `η`

. The intuition is that _{a}`η`

creates a trivial expression from a value of type _{a}`a`

. The first coherence condition that makes the algebra compatible with the monad ensures that evaluating this expression using the algebra whose carrier is `a`

gives us back the original value:

alg ∘ η_{a}= id_{a}

The second condition arises from the fact that there are two ways of evaluating the doubly nested expression `m (m a)`

. We can first apply `μ`

to flatten the expression, and then use the evaluator of the algebra; or we can apply the lifted evaluator to evaluate the inner expressions, and then apply the evaluator to the result. We’d like the two strategies to be equivalent:_{a}

alg ∘ μ_{a}= alg ∘ m alg

Here, `m alg`

is the morphism resulting from lifting `alg`

using the functor `m`

. The following commuting diagrams describe the two conditions (I replaced `m`

with `T`

in anticipation of what follows):

We can also express these condition in Haskell:

alg . return = id alg . join = alg . fmap alg

Let’s look at a small example. An algebra for a list endofunctor consists of some type `a`

and a function that produces an `a`

from a list of `a`

. We can express this function using `foldr`

by choosing both the element type and the accumulator type to be equal to `a`

:

foldr :: (a -> a -> a) -> a -> [a] -> a

This particular algebra is specified by a two-argument function, let’s call it `f`

, and a value `z`

. The list functor happens to also be a monad, with `return`

turning a value into a singleton list. The composition of the algebra, here `foldr f z`

, after `return`

takes `x`

to:

foldr f z [x] = x `f` z

where the action of `f`

is written in the infix notation. The algebra is compatible with the monad if the following coherence condition is satisfied for every `x`

:

x `f` z = x

If we look at `f`

as a binary operator, this condition tells us that `z`

is the right unit.

The second coherence condition operates on a list of lists. The action of `join`

concatenates the individual lists. We can then fold the resulting list. On the other hand, we can first fold the individual lists, and then fold the resulting list. Again, if we interpret `f`

as a binary operator, this condition tells us that this binary operation is associative. These conditions are certainly fulfilled when `(a, f, z)`

is a monoid.

## T-algebras

Since mathematicians prefer to call their monads `T`

, they call algebras compatible with them T-algebras. T-algebras for a given monad T in a category *C* form a category called the Eilenberg-Moore category, often denoted by C^{T}. Morphisms in that category are homomorphisms of algebras. These are the same homomorphisms we’ve seen defined for F-algebras.

A T-algebra is a pair consisting of a carrier object and an evaluator, `(a, f)`

. There is an obvious forgetful functor `U`

from C^{T}^{T} to C, which maps `(a, f)`

to `a`

. It also maps a homomorphism of T-algebras to a corresponding morphism between carrier objects in C. You may remember from our discussion of adjunctions that the left adjoint to a forgetful functor is called a free functor.

The left adjoint to `U`

is called ^{T}`F`

. It maps an object ^{T}`a`

in C to a free algebra in C^{T}. The carrier of this free algebra is `T a`

. Its evaluator is a morphism from `T (T a)`

back to `T a`

. Since `T`

is a monad, we can use the monadic `μ`

(Haskell _{a}`join`

) as the evaluator.

We still have to show that this is a T-algebra. For that, two coherence conditions must be satisified:

alg ∘ η_{Ta}= id_{Ta}

alg ∘ μ_{a}= alg ∘ T alg

But these are just monadic laws, if you plug in `μ`

for the algebra.

As you may recall, every adjunction defines a monad. It turns out that the adjunction between F^{T} and U^{T} defines the very monad `T`

that was used in the construction of the Eilenberg-Moore category. Since we can perform this construction for every monad, we conclude that every monad can be generated from an adjunction. Later I’ll show you that there is another adjunction that generates the same monad.

Here’s the plan: First I’ll show you that `F`

is indeed the left adjoint of ^{T}`U`

. I’ll do it by defining the unit and the counit of this adjunction and proving that the corresponding triangular identities are satisfied. Then I’ll show you that the monad generated by this adjunction is indeed our original monad.^{T}

The unit of the adjunction is the natural transformation:

η :: I -> U^{T}∘ F^{T}

Let’s calculate the `a`

component of this transformation. The identity functor gives us `a`

. The free functor produces the free algebra `(T a, μ`

, and the forgetful functor reduces it to _{a})`T a`

. Altogether we get a mapping from `a`

to `T a`

. We’ll simply use the unit of the monad `T`

as the unit of this adjunction.

Let’s look at the counit:

ε :: F^{T}∘ U^{T}-> I

Let’s calculate its component at some T-algebra `(a, f)`

. The forgetful functor forgets the `f`

, and the free functor produces the pair `(T a, μ`

. So in order to define the component of the counit _{a})`ε`

at `(a, f)`

, we need the right morphism in the Eilenberg-Moore category, or a homomorphism of T-algebras:

(T a, μ_{a}) -> (a, f)

Such homomorphism should map the carrier `T a`

to `a`

. Let’s just resurrect the forgotten evaluator `f`

. This time we’ll use it as a homomorphism of T-algebras. Indeed, the same commuting diagram that makes `f`

a T-algebra may be re-interpreted to show that it’s a homomorphism of T-algebras:

We have thus defined the component of the counit natural transformation `ε`

at `(a, f)`

(an object in the category of T-algebras) to be `f`

.

To complete the adjunction we also need to show that the unit and the counit satisfy triangular identites. These are:

The first one holds because of the unit law for the monad `T`

. The second is just the law of the T-algebra `(a, f)`

.

We have established that the two functors form an adjunction:

F^{T}⊣ U^{T}

Every adjunction gives rise to a monad. The round trip

U^{T}∘ F^{T}

is the endofunctor in C that gives rise to the corresponding monad. Let’s see what its action on an object `a`

is. The free algebra created by `F`

is ^{T}`(T a, μ`

. The forgetful functor _{a})`F`

drops the evaluator. So, indeed, we have:^{T}

U^{T}∘ F^{T}= T

As expected, the unit of the adjunction is the unit of the monad `T`

.

You may remember that the counint of the adjunction produces monadic muliplication through the following formula:

μ = R ∘ ε ∘ L

This is a horizontal composition of three natural transformations, two of them being identity natural transformations mapping, respectively, `L`

to `L`

and `R`

to `R`

. The one in the middle, the counit, is a natural transformation whose component at an algebra `(a, f)`

is `f`

.

Let’s calculate the component `μ`

. We first horizontally compose _{a}`ε`

after `F`

, which results in the component of ^{T}`ε`

at `F`

. Since ^{T}a`F`

takes ^{T}`a`

to the algebra `(T a, μ`

, and _{a})`ε`

picks the evaluator, we end up with `μ`

. Horizontal composition on the left with _{a}`U`

doesn’t change anything, since the action of ^{T}`U`

on morphisms is trivial. So, indeed, the ^{T}`μ`

obtained from the adjunction is the same as the `μ`

of the original monad `T`

.

## The Kleisli Category

We’ve seen the Kleisli category before. It’s a category constructed from another category *C* and a monad `T`

. We’ll call this category *C _{T}*. The objects in the Kleisli category

*C*are the objects of

_{T}*C*, but the morphisms are different. A morphism

`f`_{K}

from `a`

to `b`

in the Kleisli category corresponds to a morphism `f`

from `a`

to `T b`

in the original category. We call this morphism a Kleisli arrow from `a`

to `b`

.Composition of morphisms in the Kleisli category is defined in terms of monadic composition of Kleisli arrows. For instance, let’s compose `g`

after _{K}`f`

. In the Kleisli category we have:_{K}

f_{K}:: a -> b g_{K}:: b -> c

which, in the category *C*, corresponds to:

f :: a -> T b g :: b -> T c

We define the composition:

h_{K}= g_{K}∘ f_{K}

as a Kleisli arrow in *C*

h :: a -> T c h = μ ∘ (T g) ∘ f

In Haskell we would write it as:

h = join . fmap g . f

There is a functor `F`

from *C* to *C _{T}* which acts trivially on objects. On morphims, it maps

`f`

in *C*to a morphism in

*C*by creating a Kleisli arrow that embellishes the return value of

_{T}`f`

. Given a morphism:f :: a -> b

it creates a morphism in *C _{T}* with the corresponding Kleisli arrow:

η ∘ f

In Haskell we’d write it as:

return . f

We can also define a functor `G`

from *C _{T}* back to

*C*. It takes an object

`a`

from the Kleisli category and maps it to an object `T a`

in *C*. Its action on a morphism

`f`_{K}

corresponding to a Kleisli arrow:f :: a -> T b

is a morphism in *C*:

T a -> T b

given by first lifting `f`

and then applying `μ`

:

μ_{T b}∘ T f

In Haskell notation this would read:

G f_{T}= join . fmap f

You may recognize this as the definition of monadic bind in terms of `join`

.

It’s easy to see that the two functors form an adjunction:

F ⊣ G

and their composition `G ∘ F`

reproduces the original monad `T`

.

So this is the second adjunction that produces the same monad. In fact there is a whole category of adjunctions `Adj(C, T)`

that result in the same monad `T`

on *C*. The Kleisli adjunction we’ve just seen is the initial object in this category, and the Eilenberg-Moore adjunction is the terminal object.

## Coalgebras for Comonads

Analogous constructions can be done for any comonad `W`

. We can define a category of coalgebras that are compatible with a comonad. They make the following diagrams commute:

where `coa`

is the coevaluation morphism of the coalgebra whose carrier is `a`

:

coa :: a -> W a

and `ε`

and `δ`

are the two natural transformations defining the comonad (in Haskell, their components are called `extract`

and `duplicate`

).

There is an obvious forgetful functor `U`

from the category of these coalgebras to ^{W}*C*. It just forgets the coevaluation. We’ll consider its right adjoint `F`

.^{W}

U^{W}⊣ F^{W}

The right adjoint to a forgetful functor is called a cofree functor. `F`

generates cofree coalgebras. It assigns, to an object ^{W}`a`

in *C*, the coalgebra `(W a, δ`

. The adjunction reproduces the original comonad as the composite _{a})`F`

.^{W} ∘ U^{W}

Similarly, we can construct a co-Kleisli category with co-Kleisli arrows and regenerate the comonad from the corresponding adjunction.

## Lenses

Let’s go back to our discussion of lenses. A lens can be written as a coalgebra:

coalg_{s}:: a -> Store s a

for the functor `Store s`

:

data Store s a = Store (s -> a) s

This coalgebra can be also expressed as a pair of functions:

set :: a -> s -> a get :: a -> s

(Think of `a`

as standing for “all,” and `s`

as a “small” part of it.) In terms of this pair, we have:

coalg_{s}a = Store (set a) (get a)

Here, `a`

is a value of type `a`

. Notice that partially applied `set`

is a function `s->a`

.

We also know that `Store s`

is a comonad:

instance Comonad (Store s) where extract (Store f s) = f s duplicate (Store f s) = Store (Store f) s

The question is: Under what conditions is a lens a coalgebra for this comonad? The first coherence condition:

ε_{a}∘ coalg = id_{a}

translates to:

set a (get a) = a

This is the lens law that expresses the fact that if you set a field of the structure `a`

to its previous value, nothing changes.

The second condition:

fmap colag ∘ coalg = δ_{a}∘ coalg

requires a little more work. First, recall the definition of `fmap`

for the `Store`

functor:

fmap g (Store f s) = Store (g . f) s

Applying `fmap coalg`

to the result of `coalg`

gives us:

Store (coalg . set a) (get a)

On the other hand, applying `duplicate`

to the result of `coalg`

produces:

Store (Store (set a)) (get a)

For these two expressions to be equal, the two functions under `Store`

must be equal when acting on an arbitrary `s`

:

coalg (set a s) = Store (set a) s

Expanding `coalg`

, we get:

Store (set (set a s)) (get (set a s)) = Store (set a) s

This is equivalent to two remaining lens laws. The first one:

set (set a s) = set a

tells us that setting the value of a field twice is the same as setting it once. The second law:

get (set a s) = s

tells us that getting a value of a field that was set to `s`

gives `s`

back.

In other words, a well-behaved lens is indeed a comonad coalgebra for the `Store`

functor.

## Challenges

- What is the action of the free functor
`F :: C -> C`

on morphisms. Hint: use the naturality condition for monadic^{T}`μ`

. - Define the adjunction:
U

^{W}⊣ F^{W} - Prove that the above adjunction reproduces the original comonad.

## Acknowledgment

I’d like to thank Gershom Bazerman for helpful comments.

Next: Ends and Coends.

March 17, 2017 at 9:02 pm

With regards to the first of the triangle identities, don’t you need to show for all X in C that FX mapped to FGFX = FTX mapped to FX by (epsilon at FX) o (F.(eta at X)) is the same as the identity at FX? Isn’t the diagram given above looking instead at GFX =TX mapping to GFGFX = T.TX mapping to GFGX = TX (ie with G applied to each step)? Apologies if I’m missing something.

March 18, 2017 at 12:08 pm

@Dan: I’m afraid I wasn’t very clear explaining triangular identities in the post about adjunctions. These are identities that involve horizontal composition of natural transformations and their translation to components is not immediately obvious. Fortunately, these are just blog posts so I can go back and update them. So check out the updated post on adjunctions. The relevant formulas are: