我无能为力来扩大这个问题。但这里有一个用例:假设你有两个 monad 转换器,t
并且s
,在同一个 monad 上进行转换m
:
master :: (MonadTrans t, Monad m) => t m a b
slave :: (MonadTrans t, Monad m) => s m a b
而且我想编写master
and以便在将原语提升到andslave
时它们可以相互通信。签名可能是:m
t
s
bound :: (MonadTrans t, MonadTrans s, Monad m, Monoid a) => t m a b -> s m a b -> (...)
But what is the type of (...) ?
一个用例,用糖化符号表示:
master :: Monoid a => a -> t m a b
master a = do
a <- lift . send $ (a,False) -- * here master is passing function param to slave
... -- * do some logic with a
b <- lift . send $ (mempty,True) -- * master terminates slave, and get back result
slave :: Monoid a => (a -> b) -> s m a b
slave g = do
(a,end) <- lift receive
case end of
True -> get >>= \b -> exit b
_ -> (modify (++[g a])) >> slave g
更新:send
andreceive
是 type 的原语m
。
如果这个例子看起来很做作,或者太像协程,我深表歉意,问题的精神真的与它无关,所以请忽略所有相似之处。但主要的一点是 monadt
和s
之前不能明智地相互组合,但是在它们都包装了一些底层 monad 之后m
,它们现在可以组合并作为单个函数运行。至于组合函数的类型,我真的不确定,所以有些方向是值得赞赏的。现在,如果这种抽象已经存在而我只是不知道,那将是最好的。