行!
下面的代码是使用 ES6 语法编写的,但也可以很容易地使用 ES5 甚至更少的语法编写。ES6不要求创建“循环 x 次的机制”
如果回调中不需要迭代器,这是最简单的实现
const times = x => f => {
if (x > 0) {
f()
times (x - 1) (f)
}
}
// use it
times (3) (() => console.log('hi'))
// or define intermediate functions for reuse
let twice = times (2)
// twice the power !
twice (() => console.log('double vision'))
如果确实需要迭代器,则可以使用带有计数器参数的命名内部函数为您进行迭代
const times = n => f => {
let iter = i => {
if (i === n) return
f (i)
iter (i + 1)
}
return iter (0)
}
times (3) (i => console.log(i, 'hi'))
如果您不喜欢学习更多东西,请停止阅读这里...
但是这些东西应该让人感觉有些不对劲……
- 单个分支
if
语句很难看——在另一个分支上会发生什么?
- 函数体中的多个语句/表达式——过程问题是否混合?
- 隐式返回
undefined
— 指示不纯的、有副作用的函数
“没有更好的办法吗?”
有。让我们首先回顾一下我们最初的实现
// times :: Int -> (void -> void) -> void
const times = x => f => {
if (x > 0) {
f() // has to be side-effecting function
times (x - 1) (f)
}
}
当然,这很简单,但请注意我们如何只调用f()
而不用它做任何事情。这确实限制了我们可以多次重复的函数类型。即使我们有可用的迭代器,f(i)
也不是通用的多。
如果我们从一种更好的函数重复程序开始呢?也许可以更好地利用输入和输出的东西。
泛型函数重复
// repeat :: forall a. Int -> (a -> a) -> a -> a
const repeat = n => f => x => {
if (n > 0)
return repeat (n - 1) (f) (f (x))
else
return x
}
// power :: Int -> Int -> Int
const power = base => exp => {
// repeat <exp> times, <base> * <x>, starting with 1
return repeat (exp) (x => base * x) (1)
}
console.log(power (2) (8))
// => 256
上面,我们定义了一个通用repeat
函数,它接受一个额外的输入,用于启动单个函数的重复应用。
// repeat 3 times, the function f, starting with x ...
var result = repeat (3) (f) (x)
// is the same as ...
var result = f(f(f(x)))
实现times
与repeat
现在这很容易;几乎所有的工作都已经完成。
// repeat :: forall a. Int -> (a -> a) -> a -> a
const repeat = n => f => x => {
if (n > 0)
return repeat (n - 1) (f) (f (x))
else
return x
}
// times :: Int -> (Int -> Int) -> Int
const times = n=> f=>
repeat (n) (i => (f(i), i + 1)) (0)
// use it
times (3) (i => console.log(i, 'hi'))
由于我们的函数将i
作为输入并返回i + 1
,因此它可以有效地用作我们f
每次传递给的迭代器。
我们也修复了问题的项目符号列表
- 不再有丑陋的单分支
if
语句
- 单表达式主体表示很好分离的关注点
- 不再无用,隐式返回
undefined
JavaScript 逗号运算符,
如果您在查看最后一个示例的工作方式时遇到问题,这取决于您对 JavaScript 最古老的战斗轴之一的认识;的逗号操作符-总之,它从左至右计算表达式和返回最后计算的表达式的值
(expr1 :: a, expr2 :: b, expr3 :: c) :: c
在我们上面的例子中,我使用
(i => (f(i), i + 1))
这只是一种简洁的写作方式
(i => { f(i); return i + 1 })
尾调用优化
与递归实现一样性感,在这一点上,我推荐它们是不负责任的,因为我认为没有任何JavaScript VM支持正确的尾调用消除——babel 曾经用于转译它,但它已经“损坏;将重新实现” “状态超过一年。
repeat (1e6) (someFunc) (x)
// => RangeError: Maximum call stack size exceeded
因此,我们应该重新审视我们的实现repeat
以使其堆栈安全。
下面的代码确实使用了可变变量n
,x
但请注意,所有repeat
更改都针对函数进行了本地化——从函数外部看不到状态更改(突变)
// repeat :: Int -> (a -> a) -> (a -> a)
const repeat = n => f => x =>
{
let m = 0, acc = x
while (m < n)
(m = m + 1, acc = f (acc))
return acc
}
// inc :: Int -> Int
const inc = x =>
x + 1
console.log (repeat (1e8) (inc) (0))
// 100000000
这会让很多人说“但这不起作用!” ——我知道,放松点。我们可以使用纯表达式为常量空间循环实现 Clojure 风格的loop
/recur
接口;没有那些东西。while
在这里,我们抽象while
出我们的loop
函数——它寻找一种特殊的recur
类型来保持循环运行。当recur
遇到非类型时,循环结束并返回计算结果
const recur = (...args) =>
({ type: recur, args })
const loop = f =>
{
let acc = f ()
while (acc.type === recur)
acc = f (...acc.args)
return acc
}
const repeat = $n => f => x =>
loop ((n = $n, acc = x) =>
n === 0
? acc
: recur (n - 1, f (acc)))
const inc = x =>
x + 1
const fibonacci = $n =>
loop ((n = $n, a = 0, b = 1) =>
n === 0
? a
: recur (n - 1, b, a + b))
console.log (repeat (1e7) (inc) (0)) // 10000000
console.log (fibonacci (100)) // 354224848179262000000