问题在底部以粗体显示。
LYAH 给出了将do符号与Writermonad一起使用的示例
import Control.Monad.Writer
logNumber :: Int -> Writer [String] Int
logNumber x = writer (x, ["number " ++ show x])
multWithLog :: Writer [String] Int
multWithLog = do
a <- logNumber 3
b <- logNumber 5
return (x*y)
do可以在没有符号的情况下重写定义:
multWithLog = logNumber 3 >>= (\x ->
logNumber 5 >>= (\y ->
return (x*y)))
到目前为止,一切都很好。
之后,本书介绍tell并编辑了 的定义multWithLog如下:
multWithLog = do
a <- logNumber 3
b <- logNumber 5
tell ["something"]
return (x*y)
这又可以重写为:
multWithLog = logNumber 3 >>= (\x ->
logNumber 5 >>= (\y ->
tell ["something"] >>
return (x*y)))
然后这本书提出了一个对我来说似乎不清楚的观点,如果不是不准确的话:
最后一行很重要
return (a*b),因为表达式中最后一行do的结果是整个 do 表达式的结果。如果我们把tell作为最后一行,()将是这个do表达式的结果。我们会丢失乘法的结果。但是,日志将是相同的。
因此,我的第一个疑问来了:如果tell结果为(),那么代码不应该甚至不编译,因为()不能匹配预期的类型Int,也不能匹配除自身以外的任何其他类型();那么作者想告诉我们什么?为了使这种非基于意见的内容,自从本书编写以来,Haskell 中是否发生了一些变化,这使得上述引用的陈述不清楚/不准确?