How can we prove that the continuation monad has no valid instance of MonadFix
?
Why can't there be an instance of MonadFix for the continuation monad?
1.5k Views Asked by Petr At
2
There are 2 best solutions below
4

Well actually, it's not that there can't be a MonadFix
instance, just that the library's type is a bit too constrained. If you define ContT
over all possible r
s, then not only does MonadFix
become possible, but all instances up to Monad
require nothing of the underlying functor :
newtype ContT m a = ContT { runContT :: forall r. (a -> m r) -> m r }
instance Functor (ContT m) where
fmap f (ContT k) = ContT (\kb -> k (kb . f))
instance Monad (ContT m) where
return a = ContT ($a)
join (ContT kk) = ContT (\ka -> kk (\(ContT k) -> k ka))
instance MonadFix m => MonadFix (ContT m) where
mfix f = ContT (\ka -> mfixing (\a -> runContT (f a) ka<&>(,a)))
where mfixing f = fst <$> mfix (\ ~(_,a) -> f a )
Consider the type signature of
mfix
for the continuation monad.Here's the proof that there's no pure inhabitant of this type.
As you can see the problem is that both
f
andk
expect a value of typea
as an input. However, there's no way to conjure a value of typea
. Hence, there's no pure inhabitant ofmfix
for the continuation monad.Note that you can't define
mfix
recursively either becausemfix f k = mfix ? ?
would lead to an infinite regress since there's no base case. And, we can't definemfix f k = f ? ?
ormfix f k = k ?
because even with recursion there's no way to conjure a value of typea
.But, could we have an impure implementation of
mfix
for the continuation monad? Consider the following.The question that arises is how to apply
f
tox'
. Normally, we'd do this using a recursive let expression, i.e.let x' = f x'
. However,x'
is not the return value off
. Instead, the continuation given tof
is applied tox'
. To solve this conundrum, we create an empty mutable variablem
, lazily read its valuex
, and applyf
tox
. It's safe to do so becausef
must not be strict in its argument. Whenf
eventually calls the continuation given to it, we store the resultx'
inm
and apply the continuationk
tox'
. Thus, when we finally evaluatex
we get the resultx'
.The above implementation of
mfix
for the continuation monad looks a lot like the implementation ofmfix
for theIO
monad.Note, that in the implementation of
mfix
for the continuation monad we usedreadMVar
whereas in the implementation ofmfix
for theIO
monad we usedtakeMVar
. This is because, the continuation given tof
can be called multiple times. However, we only want to store the result given to the first callback. UsingreadMVar
instead oftakeMVar
ensures that the mutable variable remains full. Hence, if the continuation is called more than once then the second callback will block indefinitely on theputMVar
operation.However, only storing the result of the first callback seems kind of arbitrary. So, here's an implementation of
mfix
for the continuation monad that allows the provided continuation to be called multiple times. I wrote it in JavaScript because I couldn't get it to play nicely with laziness in Haskell.Here's the equivalent Haskell code, sans the implementation of
mfix
.Notice that this looks a lot like the list monad.
This makes sense because after all the continuation monad is the mother of all monads. I'll leave the verification of the
MonadFix
laws of my JavaScript implementation ofmfix
as an exercise for the reader.