Monads are one particular way of structuring and sequencing computations. The bind of a monad cannot magically restructure your computation so as to happen in a more efficient way. There are two problems with the way you structure your computation.
-
When evaluating
stepN 20 0
, the result ofstep 0
will be computed 20 times. This is because each step of the computation produces 0 as one alternative, which is then fed to the next step, which also produces 0 as alternative, and so on…Perhaps a bit of memoization here can help.
-
A much bigger problem is the effect of
ContT
on the structure of your computation. With a bit of equational reasoning, expanding out the result ofreplicate 20 step
, the definition offoldrM
and simplifying as many times as necessary, we can see thatstepN 20 0
is equivalent to:(...(return 0 >>= step) >>= step) >>= step) >>= ...)
All parentheses of this expression associate to the left. That’s great, because it means that the RHS of each occurrence of
(>>=)
is an elementary computation, namelystep
, rather than a composed one. However, zooming in on the definition of(>>=)
forContT
,m >>= k = ContT $ \c -> runContT m (\a -> runContT (k a) c)
we see that when evaluating a chain of
(>>=)
associating to the left, each bind will push a new computation onto the current continuationc
. To illustrate what is going on, we can use again a bit of equational reasoning, expanding out this definition for(>>=)
and the definition forrunContT
, and simplifying, yielding:setReturn 0 `setBind` (\x1 -> step x1 `setBind` (\x2 -> step x2 `setBind` (\x3 -> ...)...)
Now, for each occurrence of
setBind
, let’s ask ourselves what the RHS argument is. For the leftmost occurrence, the RHS argument is the whole rest of the computation aftersetReturn 0
. For the second occurrence, it’s everything afterstep x1
, etc. Let’s zoom in to the definition ofsetBind
:setBind set f = foldl' (\s -> union s . f) empty set
Here
f
represents all the rest of the computation, everything on the right hand side of an occurrence ofsetBind
. That means that at each step, we are capturing the rest of the computation asf
, and applyingf
as many times as there are elements inset
. The computations are not elementary as before, but rather composed, and these computations will be duplicated many times.
The crux of the problem is that the ContT
monad transformer is transforming the initial structure of the computation, which you meant as a left associative chain of setBind
‘s, into a computation with a different structure, ie a right associative chain. This is after all perfectly fine, because one of the monad laws says that, for every m
, f
and g
we have
(m >>= f) >>= g = m >>= (\x -> f x >>= g)
However, the monad laws do not impose that the complexity remain the same on each side of the equations of each law. And indeed, in this case, the left associative way of structuring this computation is a lot more efficient. The left associative chain of setBind
‘s evaluates in no time, because only elementary subcomputations are duplicated.
It turns out that other solutions shoehorning Set
into a monad also suffer from the same problem. In particular, the set-monad package, yields similar runtimes. The reason being, that it too, rewrites left associative expressions into right associative ones.
I think you have put the finger on a very important yet rather subtle problem with insisting that Set
obeys a Monad
interface. And I don’t think it can be solved. The problem is that the type of the bind of a monad needs to be
(>>=) :: m a -> (a -> m b) -> m b
ie no class constraint allowed on either a
or b
. That means that we cannot nest binds on the left, without first invoking the monad laws to rewrite into a right associative chain. Here’s why: given (m >>= f) >>= g
, the type of the computation (m >>= f)
is of the form m b
. A value of the computation (m >>= f)
is of type b
. But because we can’t hang any class constraint onto the type variable b
, we can’t know that the value we got satisfies an Ord
constraint, and therefore cannot use this value as the element of a set on which we want to be able to compute union
‘s.