我知道如何使用仅使用“绑定”操作的 list monad来执行与 Scheme(或 Python)map
和函数等效的操作。filter
这里有一些 Scala 来说明:
scala> // map
scala> List(1,2,3,4,5,6).flatMap {x => List(x * x)}
res20: List[Int] = List(1, 4, 9, 16, 25, 36)
scala> // filter
scala> List(1,2,3,4,5,6).flatMap {x => if (x % 2 == 0) List() else List(x)}
res21: List[Int] = List(1, 3, 5)
在 Haskell 中也是一样的:
Prelude> -- map
Prelude> [1, 2, 3, 4, 5, 6] >>= (\x -> [x * x])
[1,4,9,16,25,36]
Prelude> -- filter
Prelude> [1, 2, 3, 4, 5, 6] >>= (\x -> if (mod x 2 == 0) then [] else [x])
[1,3,5]
Scheme 和 Python 也有一个经常与和reduce
组合的函数。该函数使用提供的二元函数组合列表的前两个元素,然后将结果组合到下一个元素,依此类推。计算值列表的总和或乘积的常见用途。这里有一些 Python 来说明:map
filter
reduce
>>> reduce(lambda x, y: x + y, [1,2,3,4,5,6])
21
>>> (((((1+2)+3)+4)+5)+6)
21
有什么方法可以reduce
仅使用列表单子上的绑定操作来完成此操作?如果 bind 不能自行执行此操作,那么执行此操作的最“单子”方式是什么?
如果可能,请在回答时限制/避免使用语法糖(即:do
Haskell 中的符号或 Scala 中的序列理解)。