考虑以下示例程序:
next :: Int -> Int
next i
| 0 == m2 = d2
| otherwise = 3 * i + 1
where
(d2, m2) = i `divMod` 2
loopIteration :: MaybeT (StateT Int IO) ()
loopIteration = do
i <- get
guard $ i > 1
liftIO $ print i
modify next
main :: IO ()
main = do
(`runStateT` 31) . runMaybeT . forever $ loopIteration
return ()
它只能使用get
而不是lift get
因为instance MonadState s m => MonadState s (MaybeT m)
在 MaybeT 模块中定义。
许多这样的实例以组合爆炸的方式定义。
如果我们有以下类型类,那就太好了(尽管不可能?为什么?):
{-# LANGUAGE MultiParamTypeClasses #-}
class SuperMonad m s where
lifts :: m a -> s a
让我们尝试这样定义它:
{-# LANGUAGE FlexibleInstances, ... #-}
instance SuperMonad a a where
lifts = id
instance (SuperMonad a b, MonadTrans t, Monad b) => SuperMonad a (t b) where
lifts = lift . lifts
使用lifts $ print i
而不是liftIO $ print i
工作,这很好。
但是使用lifts (get :: StateT Int IO Int)
而(get :: MaybeT (StateT Int IO) Int)
不是不起作用。
GHC (6.10.3) 给出以下错误:
Overlapping instances for SuperMonad
(StateT Int IO) (StateT Int IO)
arising from a use of `lifts'
Matching instances:
instance SuperMonad a a
instance (SuperMonad a b, MonadTrans t, Monad b) =>
SuperMonad a (t b)
In a stmt of a 'do' expression:
i <- lifts (get :: StateT Int IO Int)
我明白为什么“ instance SuperMonad a a
”适用。但为什么 GHC 认为另一个也有呢?