Haskell语言学习笔记(30)MonadCont, Cont, ContT
MonadCont 类型类
class Monad m => MonadCont m where
callCC :: ((a -> m b) -> m a) -> m a
instance MonadCont (ContT r m) where
callCC = ContT.callCC
- class Monad m => MonadCont m where
MonadState 是个类型类,它为 ContT 等封装了CPS函数的 Monad 定义了通用接口。
MonadState 只包含 callCC 一个函数。该函数为CPS函数提供了显式的流程控制功能。 - instance MonadCont (ContT r m) where
callCC = ContT.callCC
对于 ContT 这个Monad转换器来说,callCC 函数的定义由 ContT 模块来提供。注意这里点运算符的含义不是函数的合成而是受限名字。
Hackage - Where is the MonadReader implementation for ReaderT defined?
ContT Monad转换器
newtype ContT r m a = ContT { runContT :: (a -> m r) -> m r }
instance Monad (ContT r m) where
return x = ContT ($ x)
m >>= k = ContT $ \c -> runContT m (\x -> runContT (k x) c)
- newtype ContT r m a = ContT { runContT :: (a -> m r) -> m r }
ContT 类型是个 newtype,也就是对现有类型的封装。该类型有三个类型参数:内部 Monad 类型参数 m,中间结果类型参数 a 以及最终结果类型参数 r。
ContT r m a 类型封装了一个CPS函数:\k -> m r(通常形式为 \k -> k a),通过 runContT 字段可以从 ContT 类型中取出这个函数。 - instance Monad (ContT r m) where
ContT r m 是一个 Monad。
对比 Monad 类型类的定义,可知 return 函数的类型签名为:
return :: a -> ContT r m a
大致相当于 a -> (a -> m r) -> m r
而 bind 操作符的类型签名为:
(>>=) :: ContT r m a -> (a -> ContT r m b) -> ContT r m b
大致相当于 (a -> m r) -> m r -> (a -> (b -> m r) -> m r) -> (b -> m r) -> m r - return x = ContT ($ x)
return 函数把 a 类型的值 x 封装进了 ContT Monad 中。
return x = ContT $ \k -> k x
这里 x 的类型为 a, k 的类型为 a -> m r,k x 的类型为 m r,
\k -> k x 的类型为 (a -> m r) -> m r,
ContT $ \k -> k x 也就是 ContT ($ x) 的类型为 ContT r m a。 - m >>= k = ContT $ \c -> runContT m (\x -> runContT (k x) c)
对比函数签名,可知 m 的类型为 ContT r m a,大致相当于 (a -> m r) -> m r。
而 k 的类型为 a -> ContT r m b,大致相当于 a -> (b -> m r) -> m r。
x 的类型为 a,c 的类型为 b -> m r。
k x 的类型 ContT r m b,
runContT (k x) 的类型为 (b -> m r) -> m r,
runContT (k x) c 的类型为 m r。
\x -> runContT (k x) c 的类型为 a -> m r,
runContT m 的类型为 (a -> m r) -> m r,
runContT m (\x -> runContT (k x) c) 的类型为 m r。
\c -> runContT m (\x -> runContT (k x) c) 的类型为 (b -> m r) -> m r,
ContT $ \c -> runContT m (\x -> runContT (k x) c) 的类型为 ContT r m b。 - m >>= k = ContT $ \c -> runContT m (\x -> runContT (k x) c)
bind 操作符组合两个封装了CPS函数的 Cont Monad,计算结果仍然是一个封装了CPS函数的 Cont Monad。
这里假设 m = ContT c1 where c1 f1 = f1 a,k = \a -> ContT c2 where c2 f2 = f2 b。
bind 操作符具体计算流程如下:
首先通过 runContT m 把封装在 bind 操作符的左操作数 m 中的CPS函数 c1 取出来,
c1 函数将把类型为 a 的运算结果 x 传递给 f1,也就是 (\x -> runContT (k x) c) 这个函数,得到 runContT (k x) c 这个运算结果。
然后 runContT (k x) 再把 bind 操作符的右操作数 k 中所封装的CPS函数 c2 取出来。
c2 函数将把类型为 b 的运算结果 y 传递给 f2, 也就是最外层的函数 c,得到最终运算结果 c y。
最后通过 ContT $ \c -> c y 这一形式,把计算结果重新封装进了 ContT Monad。
证明ContT符合Monad法则:
1. return a >>= f ≡ f a
return a >>= f
≡ ContT (\k -> k a) >>= f
≡ ContT $ \c -> runContT (ContT (\k -> k a)) (\x -> runContT (f x) c)
≡ ContT $ \c -> (\k -> k a) (\x -> runContT (f x) c)
≡ ContT $ \c -> (\x -> runContT (f x) c) a
≡ ContT $ \c -> runContT (f a) c
≡ ContT $ runContT (f a)
≡ f a
2. m >>= return ≡ m
m >>= return
≡ ContT $ \c -> runContT m (\x -> runContT (return x) c)
≡ ContT $ \c -> runContT m (\x -> runContT (ContT $ \k -> k x) c)
≡ ContT $ \c -> runContT m (\x -> (\k -> k x) c)
≡ ContT $ \c -> runContT m (\x -> c x)
≡ ContT $ \c -> runContT m c
≡ ContT $ runContT m
≡ m
3. (m >>= f) >>= g ≡ m >>= (\x -> f x >>= g)
(m >>= f) >>= g
≡ (ContT $ \c -> runContT m (\x -> runContT (f x) c)) >>= g
≡ ContT $ \c -> runContT (ContT $ \c -> runContT m (\x -> runContT (f x) c)) (\x -> runContT (g x) c)
≡ ContT $ \c -> (\c -> runContT m (\x -> runContT (f x) c)) (\x -> runContT (g x) c)
≡ ContT $ \c -> runContT m (\x -> runContT (f x) (\x -> runContT (g x) c))
m >>= (\x -> f x >>= g)
≡ ContT $ \c -> runContT m (\x -> runContT ((\x -> f x >>= g) x) c)
≡ ContT $ \c -> runContT m (\x -> runContT (f x >>= g) c)
≡ ContT $ \c -> runContT m (\x -> runContT (ContT $ \c -> runContT (f x) (\x -> runContT (g x) c)) c)
≡ ContT $ \c -> runContT m (\x -> (\c -> runContT (f x) (\x -> runContT (g x) c)) c)
≡ ContT $ \c -> runContT m (\x -> runContT (f x) (\x -> runContT (g x) c))
lift liftIO 函数
instance MonadTrans (ContT r) where
lift m = ContT (m >>=)
instance (MonadIO m) => MonadIO (ContT r m) where
liftIO = lift . liftIO
lift m 将封装在内部 Monad m 的值封装进了 ContT Monad 之中。
lift m = ContT $ \k -> m >>= k
这里 m 的类型为 m a, k 的类型为 a -> m r,m >> k 的类型为 m r,
\k -> m >>= k 的类型为 (a -> m r) -> m r,
ContT $ \k -> m >>= k 也就是 ContT (m >>=) 的类型为 ContT r m a。
证明 ContT 中 lift 函数的定义符合 lift 的法则。
1. lift . return ≡ return
lift . return $ a
≡ lift (m a)
≡ ContT (m a >>=)
≡ ContT $ \k -> m a >>= k
≡ ContT $ \k -> k a
≡ return a
2. lift (m >>= f) ≡ lift m >>= (lift . f)
假设 m = n a 并且 f a = n b
于是 m >>= f = n b
lift (m >>= f)
≡ lift (n b)
≡ ContT (n b >>=)
≡ ContT $ \k -> n b >>= k
≡ ContT $ \k -> k b
≡ return b
lift m >>= (lift . f)
≡ ContT (n a >>=) >>= \x -> ContT (f x >>=)
≡ ContT (\k -> k a) >>= \x -> ContT (\k -> f x >>= k)
≡ ContT $ \c -> runContT $ ContT (\k -> k a) (\x -> runContT $ ContT (\k -> f x >>= k) c)
≡ ContT $ \c -> (\k -> k a) (\x -> (\k -> f x >>= k) c)
≡ ContT $ \c -> (\x -> (\k -> f x >>= k) c) a
≡ ContT $ \c -> (\k -> f a >>= k) c
≡ ContT $ \c -> (\k -> n b >>= k) c
≡ ContT $ \c -> (\k -> k b) c
≡ ContT $ \c -> c b
≡ return b
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> f a = if even a then Just (a `div` 2) else Nothing
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> c = lift . f :: Int -> ContT Int Maybe Int
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> runContT (c 3) return
Nothing
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> runContT (c 4) f
Just 1
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> c = lift getLine :: ContT r IO String
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> runContT c print
abc
"abc"
callCC 函数
callCC :: ((a -> ContT r m b) -> ContT r m a) -> ContT r m a
callCC f = ContT $ \c -> runContT (f (\x -> ContT $ \_ -> c x)) c
CallCC 是 Call With Current Continuation(对当前延续函数进行调用)的缩写,它为 ContT Monad 提供了退出的手段。
CallCC 有一个参数 f,CallCC f 返回一个 ContT Monad。
f 的类型是 (a -> ContT r m b) -> ContT r m a,也就是说 f 本身是个函数,它的返回值类型和 CallCC f 相同,都是一个ContT Monad。
f 通常采用 \exit -> do ... 这种形式。
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> runContT (callCC (\exit -> exit 1) :: ContT Int Maybe Int) return
Just 1
Prelude Control.Monad.Trans.Cont Control.Monad.Trans> runContT (callCC (\exit -> do{exit 1; return 2}) :: ContT Int Maybe Int) return
Just 1
也就是说只要 CallCC 的参数采用 \exit -> do {...; exit a; ...} 这种形式,
那么该函数所返回的 ContT Monad 将无视 exit a 后面的处理流程,无条件地将 a 传递给外围函数。
下面看看 CallCC 函数是如何做到这一点的。
- callCC :: ((a -> ContT r m b) -> ContT r m a) -> ContT r m a
callCC f = ContT $ \c -> runContT (f (\x -> ContT $ \_ -> c x)) c
ContT $ \c -> runContT (f (\x -> ContT $ \_ -> c x)) c 类型为 ContT r m a
\c -> runContT (f (\x -> ContT $ \_ -> c x)) c 类型为 (a -> m r) -> m r
c 的类型为 a -> m r,
runContT (f (\x -> ContT $ \_ -> c x)) 的类型为 (a -> m r) -> m r,
f (\x -> ContT $ \_ -> c x) 的类型为 ContT r m a,
f 的类型为 (a -> ContT r m b) -> ContT r m a,
\x -> ContT $ \_ -> c x 的类型为 a -> ContT r m b,x 的类型为 a,
ContT $ \_ -> c x 的类型为 ContT r m b,
\_ -> c x 的类型为 (b -> m r) -> m r,
_ 的类型为 b -> m r,
c x 的类型为 m r。 - callCC f = ContT $ \c -> runContT (f (\x -> ContT $ \_ -> c x)) c
这里假设 f = \exit -> exit a >>= k。
callCC f where f = \exit -> exit a >>= k
= ContT $ \c -> runContT (f (\x -> ContT $ \_ -> c x)) c
= ContT $ \c -> runContT ((\exit -> exit a >>= k) (\x -> ContT $ \_ -> c x)) c
= ContT $ \c -> runContT ((\x -> ContT $ \_ -> c x) a >>= k) c
= ContT $ \c -> runContT ((ContT $ \_ -> c a) >>= k) c - m >>= k = ContT $ \c -> runContT m (\x -> runContT (k x) c)
(ContT $ \_ -> c a) >>= k
= ContT $ \c -> runContT (ContT $ \_ -> c a) (\x -> runContT (k x) c)
= ContT $ \c -> (\_ -> c a) (\x -> runContT (k x) c)
= ContT $ \c -> c a - callCC f where f = \exit -> exit a >>= k
= ContT $ \c -> runContT ((ContT $ \_ -> c a) >>= k) c
= ContT $ \c -> runContT (ContT $ \c -> c a) c
= ContT $ \c -> (\c -> c a) c
= ContT $ \c -> c a - callCC f = ContT $ \c -> runContT (f (\x -> ContT $ \_ -> c x)) c
这里假设 f = \exit -> exit a >>= k >>= h。
f = \exit -> exit a >>= k >>= h
= \exit -> exit a >>= (\x -> k x >> h)
= \exit -> exit a >>= k' where k' = \x -> k x >> h
callCC f where f = \exit -> exit a >>= k >>= h
= callCC f where f = \exit -> exit a >>= k' where k' = \x -> k x >> h
= ContT $ \c -> c a
从以上推导过程可以看出 CallCC 的参数如果采用 \exit -> do {...; exit a; ...} 这种形式,
该函数所返回的 ContT Monad 确实会无视 exit a 后面的处理流程,无条件地将 a 传递给外围函数。
ContT Monad转换器的其他函数
evalContT :: (Monad m) => ContT r m r -> m r
evalContT m = runContT m return
mapContT :: (m r -> m r) -> ContT r m a -> ContT r m a
mapContT f m = ContT $ f . runContT m
withContT :: ((b -> m r) -> (a -> m r)) -> ContT r m a -> ContT r m b
withContT f m = ContT $ runContT m . f
resetT :: (Monad m) => ContT r m r -> ContT r' m r
resetT = lift . evalContT
shiftT :: (Monad m) => ((a -> m r) -> ContT r m r) -> ContT r m a
shiftT f = ContT (evalContT . f)
liftLocal :: (Monad m) => m r' -> ((r' -> r') -> m r -> m r) ->
(r' -> r') -> ContT r m a -> ContT r m a
liftLocal ask local f m = ContT $ \c -> do
r <- ask
local f (runContT m (local (const r) . c))
Cont Monad
cont :: ((a -> r) -> r) -> Cont r a
cont f = ContT (\c -> Identity (f (runIdentity . c)))
runCont :: Cont r a -> (a -> r) -> r
runCont m k = runIdentity (runContT m (Identity . k))
evalCont :: Cont r r -> r
evalCont m = runIdentity (evalContT m)
mapCont :: (r -> r) -> Cont r a -> Cont r a
mapCont f = mapContT (Identity . f . runIdentity)
withCont :: ((b -> r) -> (a -> r)) -> Cont r a -> Cont r b
withCont f = withContT ((Identity .) . f . (runIdentity .))
reset :: Cont r r -> Cont r' r
reset = resetT
shift :: ((a -> r) -> Cont r r) -> Cont r a
shift f = shiftT (f . (runIdentity .))
Cont Monad 是 ContT Monad(转换器) 的一个特例。
应用实例
import Control.Monad.Trans.Cont
import Control.Monad (when)
add :: Int -> Int -> Int
add x y = x + y
square :: Int -> Int
square x = x * x
add_cont :: Int -> Int -> Cont r Int
add_cont x y = return (add x y)
square_cont :: Int -> Cont r Int
square_cont x = return (square x)
pythagoras_cont :: Int -> Int -> Cont r Int
pythagoras_cont x y = do
x_squared <- square_cont x
y_squared <- square_cont y
add_cont x_squared y_squared
pythagoras_cont' :: Int -> Int -> Cont r Int
pythagoras_cont' x y = callCC $ \exit -> do
when (x < 0 || y < 0) $ exit (-1)
x_squared <- square_cont x
y_squared <- square_cont y
add_cont x_squared y_squared
main = do
runCont (pythagoras_cont 3 4) print -- 25
runCont (pythagoras_cont' 3 4) print -- 25
runCont (pythagoras_cont' (-3) 4) print -- -1
How and why does the Haskell Cont monad work?
Haskell语言学习笔记(30)MonadCont, Cont, ContT的更多相关文章
- Haskell语言学习笔记(88)语言扩展(1)
ExistentialQuantification {-# LANGUAGE ExistentialQuantification #-} 存在类型专用的语言扩展 Haskell语言学习笔记(73)Ex ...
- Haskell语言学习笔记(79)lambda演算
lambda演算 根据维基百科,lambda演算(英语:lambda calculus,λ-calculus)是一套从数学逻辑中发展,以变量绑定和替换的规则,来研究函数如何抽象化定义.函数如何被应用以 ...
- Haskell语言学习笔记(69)Yesod
Yesod Yesod 是一个使用 Haskell 语言的 Web 框架. 安装 Yesod 首先更新 Haskell Platform 到最新版 (Yesod 依赖的库非常多,版本不一致的话很容易安 ...
- Haskell语言学习笔记(20)IORef, STRef
IORef 一个在IO monad中使用变量的类型. 函数 参数 功能 newIORef 值 新建带初值的引用 readIORef 引用 读取引用的值 writeIORef 引用和值 设置引用的值 m ...
- Haskell语言学习笔记(39)Category
Category class Category cat where id :: cat a a (.) :: cat b c -> cat a b -> cat a c instance ...
- Haskell语言学习笔记(44)Lens(2)
自定义 Lens 和 Isos -- Some of the examples in this chapter require a few GHC extensions: -- TemplateHas ...
- Haskell语言学习笔记(72)Free Monad
安装 free 包 $ cabal install free Installed free-5.0.2 Free Monad data Free f a = Pure a | Free (f (Fre ...
- Haskell语言学习笔记(38)Lens(1)
Lens Lens是一个接近语言级别的库,使用它可以方便的读取,设置,修改一个大的数据结构中某一部分的值. view, over, set Prelude> :m +Control.Lens P ...
- Haskell语言学习笔记(92)HXT
HXT The Haskell XML Toolbox (hxt) 是一个解析 XML 的库. $ cabal install hxt Installed hxt-9.3.1.16 Prelude&g ...
随机推荐
- bzoj 4566 [Haoi2016]找相同字符——广义后缀自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4566 每个后缀结尾处 ct[ ] = 1 ,按拓扑序 dp 一下就能求出 right 集合的 ...
- Angular 4 投影
1.创建工程 ng new demo4 2. 创建子组件 ng g component child 3.子组件html定义 <div class="wrapper"> ...
- Linux内存管理和应用
[作者:byeyear.首发于cnblogs,转载请注明.联系:east3@163.com] 本文对Linux内存管理使用到的一些数据结构和函数作了简要描述,而不深入到它们的内部.对这些数据结构和函数 ...
- python接口自动化20-requests获取响应时间(elapsed)与超时(timeout)
前言 requests发请求时,接口的响应时间,也是我们需要关注的一个点,如果响应时间太长,也是不合理的. 如果服务端没及时响应,也不能一直等着,可以设置一个timeout超时的时间 关于reques ...
- unittest框架进坑系列_(含selenium数据分离的坑)
1.测试用例的执行顺序 有默认的顺序的,不是按你自己的排列执行,注意. 进坑原因,没有先执行制造变量的测试用例,导致其他用例无法找到变量值 2.数据分离的坑 在控制层 有函数嵌套,2个函数都必须带se ...
- JavaScript-Tool:Moment.js
ylbtech-JavaScript-Tool:Moment.js Parse, validate, manipulate, and display dates and times in JavaSc ...
- web项目除了业务还需要关注的点
1:安全性,不允许访问外网,访问外网通过反向代理的方式. 2:安全性,和外网交互的时候,需要CA证书,基于SSL协议的证书 3:日志,生产上通常会关闭某些日志,所以,允许出现的日志就显得至关重要了. ...
- php变量详细讲解
变量是用于存储信息的"容器". 定义一个变量的语法: $变量名 = 值; 使用变量的例子: <?php $x=5; $y=6; $z=$x+$y; echo $z; ?> ...
- ORM( ORM查询13种方法3. 单表的双下划线的使用 4. 外键的方法 5. 多对多的方法 ,聚合,分组,F查询,Q查询,事务 )
必知必会13条 <1> all(): 查询所有结果 <2> get(**kwargs): 返回与所给筛选条件相匹配的对象,返回结果有且只有一个,如果符合筛选条件的对象超过一个或 ...
- 网络层-IP地址
以下内容是IPv4 IP地址长度32位,Java里面一个int的长度,总共分为5类IP地址 1:分类编址 A类IP地址0开头: A类有31个位置可以变化,总数是2^31个, [(0 ...