这是个简单的值:
data:image/s3,"s3://crabby-images/ea24b/ea24b9e2388ceea538dc8c108dba3ccfb989179a" alt=""
我们都知道怎么加一个函数应用到这个值上边:
data:image/s3,"s3://crabby-images/4718f/4718f02f4b46a79e84495dd09d425e181fb214eb" alt=""
很简单了. 我们来扩展一下, 让任意的值是在一个上下文当中. 现在的情况你可以想象一个可以把值放进去的盒子:
data:image/s3,"s3://crabby-images/a50b9/a50b9532fd2d9822b2e2440f023d7262952f57b3" alt=""
现在你把一个函数应用到这个值的时候,根据其上下文你会得到不同的结果. 这就是 Functor, Applicative, Monad, Arrow 之类概念的基础.Maybe数据类型定义了两种相关的上下文:
data:image/s3,"s3://crabby-images/3ba5a/3ba5a039f0f9eeb0ee79652d9204d6c0b9adc38b" alt=""
很快我们会看到对一个Justa和一个Nothing来说函数应用有何不同. 首先我们来说 Functor!
Functor
当一个值被封装在一个上下文里, 你就不能拿普通函数来应用:
data:image/s3,"s3://crabby-images/02d7b/02d7b9cb297ad6c40f319ade586668b4abc3ccc5" alt=""
就在这里fmap出现了.fmapis from the street,fmapis hip to contexts.fmap知道怎样将一个函数应用到一个带有上下文的值. 你可以对任何一个类型为Functor的类型使用fmap.
比如说, 想一下你想把(+3)应用到Just2. 用fmap:
>fmap(+3)(Just2)Just5
这是在幕后所发生的:
data:image/s3,"s3://crabby-images/6ec8f/6ec8f8e17f80365818f6d4c2e80ef6e29936be5b" alt=""
Bam!fmap告诉了我们那是怎么做到的!
So then you’re like, 好吧fmap, 请应用(+3)到一个Nothing?
data:image/s3,"s3://crabby-images/c86eb/c86eb4cc163cb086be6d17e8fb3b11fef81ff721" alt=""
>fmap(+3)NothingNothing
就像 Matrix 里的 Morpheus,fmap就是知道要做什么; 你从Nothing开始, 那么你再由Nothing结束!fmap是禅. So now you’re all like,准确说究竟什么是 Functor?嗯, Functor 就是任何能用fmap操作的数据类型. 因此Maybe是个 functor. 而且我们很快会看到, list 也是 functor.
这样上下文存在就有意义了. 比如, 这是在没有Maybe的语言里你操作一个数据库记录的方法:
post=Post.find_by_id(1)ifpostreturnpost.titleelsereturnnilend
但用 Haskell:
fmap(getPostTitle)(findPost1)
如果findPost返回一条 post, 我们就通过getPostTitle得到了 title. 如果返回的是Nothing, 我们加e得到Nothing! 非常简洁, huh?<$>是fmap的中缀表达式版本, 所以你经常是会看到:
getPostTitle<$>(findPost1)
另一个例子: 当你把函数应用到 list 时发生了什么?
data:image/s3,"s3://crabby-images/94b06/94b066991a7d647a10204f77d030bf5ad695e8a6" alt=""
List 仅仅是另一种让fmap以不同方式应用函数的上下文!
Okay, okay, 最后一个例子: 你把一个函数应用到另一个函数时会发生什么?
fmap(+3)(+1)
这是个函数:
data:image/s3,"s3://crabby-images/b22c4/b22c48e3052d0b3f7355f4cf05257d5c0a981a93" alt=""
这是一个函数应用到另一个函数上:
data:image/s3,"s3://crabby-images/2583c/2583ca8188d36cc88f6e0434afd5283967862aa6" alt=""
结果就是又一个函数!
>importControl.Applicative>let foo=(+3)<$>(+2)>foo1015
这就是函数复合! 就是说,f<$>g==f.g!
注意:目前为止我们做的是将上下文当作是一个容纳值的盒子. But sometimes the box analogy wears a little thin. 特别要记住: 盒子是有效的记忆图像, 然呵又是你并没有盒子. 有时你的 “盒子” 是个函数.
Applicative
Applicative 把这带到了一个新的层次. 借助 applicative, 我们的 values 就被封装在了上下文里, 就像 Functor:
data:image/s3,"s3://crabby-images/987e2/987e268b4ed924ebb91d8c5aef95da60255fd85c" alt=""
而我们的函数也被封装在了上下文里!
data:image/s3,"s3://crabby-images/b7103/b71031cfdf430f2bced1f12e2fba20efc8e84de3" alt=""
Yeah. Let that sink in. Applicative 并不是开玩笑.Control.Applicative定义了<*>, 这个函数知道怎样把封装在上下文里的函数应用到封装在上下文里的值:
data:image/s3,"s3://crabby-images/6271c/6271c98cbf465649d1685fde9bc0c538e2e65a98" alt=""
也就是:
Just(+3)<*>Just2==Just5
使用<*>能带来一些有趣的情形. 比如:
>[(*2),(+3)]<*>[1,2,3][2,4,6,4,5,6]
data:image/s3,"s3://crabby-images/c334d/c334de3be067a501a91f69e64ece1115a13d3808" alt=""
这里有一些是你能用 Applicative 做, 而无法用 Functor 做到的. 你怎么才能把需要两个参数的函数应用到两个封装的值上呢?
>(+)<$>(Just5)Just(+5)>Just(+5)<$>(Just4)ERROR???WHAT DOES THIS EVEN MEAN WHY IS THE FUNCTION WRAPPED IN A JUST
Applicative:
>(+)<$>(Just5)Just(+5)>Just(+5)<*>(Just3)Just8
Applicative把Functor推到了一边. “大腕儿用得起任意个参数的函数,” 他说. “用<$>和<*>武装之后, 我可以接受需要任何个未封装的值的函数. 然后我传进一些封装过的值, 再我就得到一个封装的值的输出! AHAHAHAHAH!”
>(*)<$>Just5<*>Just3Just15
data:image/s3,"s3://crabby-images/0e56d/0e56d18ef619d1b110e0c332334676fb149d5d54" alt=""
一 applicative 看着一 functor 应用一函数
还有啦! 有一个叫做liftA2的函数也做一样的事:
>liftA2(*)(Just5)(Just3)Just15
Monad
如何学习 Monad:
拿个计算机科学的 PhD.
把她抛在一边, 因为这个章节里你用不到她!
Monads add a new twist.
Functor 应用函数到封装过的值:
data:image/s3,"s3://crabby-images/13502/13502270c71b522f1c72a9777dbd6a418ba3e8bc" alt=""
Applicative 应用封装过的函数到封装过的值:
data:image/s3,"s3://crabby-images/88996/88996739943c26f6dc90afc95305da2b38790f9f" alt=""
Monads 应用会返回封装过的值的函数到封装过的值. Monad 有个>>=(念做 “bind”) 来做这个.
一起看个例子. Good ol’Maybeis a monad:
data:image/s3,"s3://crabby-images/254ba/254ba7e055f396e1371d151c12f0eb64f775a5e0" alt=""
Just a monad hanging out
假定half是仅对偶数可用的函数:
half x=ifeven xthenJust(x`div`2)elseNothing
data:image/s3,"s3://crabby-images/da611/da611a09457d1275ab8b91b35b01acbe56d0d4b8" alt=""
我们给它传入一个封装过的值会怎样?
data:image/s3,"s3://crabby-images/4399a/4399ae49cf0cb724d10c08ce9028965d36e0e6e0" alt=""
我们要用到>>=, 用来强推我们封装过的值到函数里去. 这是>>=的照片:
data:image/s3,"s3://crabby-images/4c92d/4c92de182aee845dae7b929b2e5c99b38e87170a" alt=""
它怎么起作用的:
>Just3>>=halfNothing>Just4>>=halfJust2>Nothing>>=halfNothing
其中发生了什么?
data:image/s3,"s3://crabby-images/c3f26/c3f26ac0531dbdd4db0b6220a32c7137f646d774" alt=""
如果你传进一个Nothing就更简单了:
data:image/s3,"s3://crabby-images/085cf/085cfd5f4a7bf7e226894f25aefc50174532f670" alt=""
data:image/s3,"s3://crabby-images/73555/735550c5b7dab97cadbd765371f1c1807aee1582" alt=""
酷! 我们来看另一个例子: 那个IOmonad:
data:image/s3,"s3://crabby-images/d34c7/d34c71244a0c317392b511490f6dca32aaf0e785" alt=""
明确的三个函数.getLine获取用户输入而不接收参数:
data:image/s3,"s3://crabby-images/6f759/6f7592cd985b832d842fd31c8867ebc256c742cf" alt=""
getLine::IOString
readFile接收一个字符串 (文件名) 再返回文件的内容:
data:image/s3,"s3://crabby-images/c2476/c2476fc7bfbafa3d115800fac9e8728aff87d07f" alt=""
readFile::FilePath->IOString
putStrLn接收一个字符串打印:
data:image/s3,"s3://crabby-images/1901f/1901fa4aa8a96a784d406f08bc9e9f849fa3f22c" alt=""
putStrLn::String->IO()
这三个函数接收一个常规的值 (或者不接收值) 返回一个封装过的值. 我们可以用>>=把一切串联起来!
data:image/s3,"s3://crabby-images/49317/49317335464a178c092ca9b9dc0f4310f49ae734" alt=""
getLine>>=readFile>>=putStrLn
Aw yeah! 我们不需要在取消封装和重新封装 IO monad 的值上浪费时间.>>=为我们做了那些工作!
Haskell 还为 monad 提供了语法糖, 叫做do表达式:
foo=dofilename<-getLine
contents<-readFile filename
putStrLn contents
结论
data:image/s3,"s3://crabby-images/b8ebf/b8ebf4972603442779d3e0efd61d088d15265088" alt=""
functor:通过fmap或者<$>应用是函数到封装过的值
applicative:通过<*>或者liftA应用封装过的函数到封装过的值
网友评论