-
Notifications
You must be signed in to change notification settings - Fork 54
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
在 JavaScript 中用匿名函数(箭头函数)写出递归的方法 #7
Comments
👍 |
应为
|
@JokerQyou 感谢指出。已改正。 |
//定义时就工厂化,生产出阶乘函数
let factory = self => n => n < 2 ? 1 : n * self(n - 1) 我觉得这里不太对,这里说的factory是一个生成lambd函数的工厂,而真正需要形成递归的是 其生成的lambd函数,但是 这里的self指的是factory而不是 其生成的lambda函数,及这里的self需要替换成 需要递归的lambda函数——>最简单的方式: let factory = self => n => n<2 ? 1 : n*self(self)(n-1)
//这样Y就变得非常简单了,
let Y = f => f(f) 还有,我有一样东西想不明白,为什么要执着于用lambda函数做递归? var f1 = function (s) {
return function (n) {
return n<2 ? 1: n*s(s)(n-1)
}
}
//your function
var f2 = function (s) {
return function (n) {
return n<2 ? 1: n*s(n-1)
}
}
var simpleY = function (f) {
return f(f);
}
//your Y
var Y = function (factory) {
var magic = function (self) {
return function (n) {
return factory(self(self))(n)
}
}
return magic(magic)
}
var e1 = simpleY(f1)
var e2 = Y(f2)
function normal(n){
return n < 2 ? 1: n*normal(n-1)
}
function player(f, n, loop){
for(var i=0;i< loop;i++){
f(n)
}
}
var start = new Date()
player(normal, 1000, 10000)
var end1 = new Date()
player(e1, 1000, 10000)
var end2 = new Date()
player(e2, 1000, 10000)
var end3 = new Date()
console.log(end1 - start)
console.log(end2 - end1)
console.log(end3 - end2)
//result:
//136
//558
//1382
//我的电脑低性能,求原谅 实在不明,求解答 |
@MephistoMMM 你的思路非常不错。 //这个 self 指的是阶乘函数自身,只是一个语义提示
let factory = self => n => n < 2 ? 1 : n * self(n - 1)
//也可以换成这样
let factory = fact=> n => n < 2 ? 1 : n * fact(n - 1) //这样做,很难反映出 fact 就是 factory 生产出来的。
//这样Y就变得非常简单了
let Y = f => f(f)
//但这样 factory 就变得复杂了
let factory = self => n => n < 2 ? 1 : n * self(self)(n-1) 至于性能测试,这篇文章的内容,不是为了投入实际生产,不是一个实用技巧。 更像是一个科普资讯,展现 js 里的匿名函数跟函数式语言里的 lambda 表达式的关系,以及跟计算机里的 lambda 跟图灵机,跟数学里的 lambda 演算,跟哥德尔不完备定理,跟康托尔的集合论之间存在的千丝万缕的联系。 这是 JavaScript 所处的大背景,这篇文章是为了让更多人能在在这么个背景下,重新审视这门语言。 |
//这样Y就变得非常简单了
let Y = f => f(f)
//但这样 factory 就变得复杂了
let factory = self => n => n < 2 ? 1 : n * self(self)(n-1) 我觉得这里的问题好像不是谁简单谁复杂,而是这样的 |
发现了,还真是这样~~ |
@myst729 对的~文章的目的不是简化 |
好文,找了很多篇讲 YC 的,这篇最好懂。多谢博主。
手动转化为 es6 箭头函数,结果如下:
这和 λ演算 的写法不太一样啊,λ演算是
问题就出在 |
@iwestlin |
let fact = n => n < 2 ? 1 : n * fact(n - 1)
fact(4) // => 24 看,我们用匿名函数实现了递归,全剧终...... 不,那只是 JS 引擎给我们的语法糖。实际上,所谓的「用 lambda 表达式写出递归」,不能在 lambda 定义完成之前直接引用自身。我们做如下假设: let fact = n => n < 2 ? 1 : n * fact(n - 1) //抛出错误: lambda 表达式引用错误 请问,“那只是 JS 引擎给我们的语法糖”。具体是什么语法糖。我也好奇 |
前言
今天看
Mozilla
出品的 ES6 In Depth ,看到 Arrow functions(中文翻译),其中一段让人讶异。里面提到λ (lambda)表达式、阿隆佐·邱奇(Alonzo Church)、阿兰·图灵(Alan Turing),这些耳熟能详的名词原来与我们写 JavaScript 的人这么近,这激发了我极大的探索兴趣。
最后搜索到刘未鹏2006年的一篇文章《康托尔、哥德尔、图灵——永恒的金色对角线(rev#2)》,奇妙的从 ES2015 延伸到了计算机的起源以及现代数学史的开端。
原来我们轻易写下的每一个匿名函数,里面都蕴涵简单而又玄妙的数学原理。
原来用匿名函数实现的递归,动用了如此深刻的数学法则。
希望每个前端工程师都能认真阅读刘未鹏的文章,理解
Y Combinator
的JavaScript
实现,对这门正在越变越好的语言抱以更多的敬畏之情,写起 ES2015 来或许有更好的编程体验。注:本文部分代码将用 ES2015 编写,要跑起来可能得先用Babel编译为 ES5。
正文
我们用递归的方式实现阶乘函数,并且从朴素思路出发,最后一步步抵达
Y Combinator
。首先,用最简单的命名函数递归方式,如下:
第二种方式,用变量缓存匿名函数的值:
看,我们用匿名函数实现了递归,全剧终......
不,那只是 JS 引擎给我们的语法糖。实际上,所谓的「用 lambda 表达式写出递归」,不能在 lambda 定义完成之前直接引用自身。我们做如下假设:
在这个基础上,继续探索我们的话题。
如果 lambda 表达式不能直接在函数体内显示引用自身,那么我们就得隐式地调用自身;因为我们不是用循环来模拟递归,我们就是要让 lambda 表达式反复执行一段相同代码。
其中一个策略是,将 lambda 表达式作为参数之一传入其自身。(函数也是值)
OK,我们现在的确实现了具有递归效果的 lambda 表达式,但是,太难看了。没有人希望自己的阶乘函数有多余的参数,我们的目标是,
fact(n)
。为了达到参数纯净化目的,我们可以包裹一层工厂函数,封装肮脏的冗余传参行为。
虽然现在我们达到了在调用时参数纯净化的目标,但仍有些不美。定义
fact
时,我们还在self(self, n - 1)
, 方式不够直观,我们期望能用下面的方式代替。在函数被定义之后,我们才拿到其引用;也就是说,不可能在生产/创建一个函数时,把它自己传参进去。也就是说,对于上面的工厂函数
factory
而言,self === factory(self)
永远不可能为真。不过,没关系。我们有软件工程里的黄金定律:既然无法让一个阶乘函数反复调用自身,那就让
factory
在需要时反复生产出虽然不是同一个,但效果等价的、新的阶乘函数。我们设想有以下特征的Y
函数:在知道
Y
函数的功能与行为后,我们再根据已知条件,把它构造出来。首先,
Y
函数一定返回阶乘函数,那么它的可能形式如下,其次,Y 一定调用了
factory
函数两次以上magic
函数从factory
取出新的阶乘函数,作为参数又传入factory
,这样创建出来的阶乘函数,里面的 self 就是另一个阶乘函数。到这里,我们只需要探究 magic 应该是什么代码形式。
可惜,上面复用 magic 函数,也只是语法糖,我们不能在 magic 定义完成前显式引用它。
诶?
那么就再增加中间层,隐式引用呗。说做就做。
惊!!,我们竟然成功了。虽然我们不知道
magic
魔术函数为什么是那样,但是,我们把它构造了出来。同时,我们注意到,
magic
的factory
参数,好像没有存在的必要了,因为作用域内只存在唯一一个factory
。神奇。
magic
魔术函数果然很魔术,在外部magic(magic)
自己调用自己, 在内部self(self)
,就实现了递归?同时,我们又注意到一点,
n => factory(magic(magic))(n)
的形式跟n => factory(self(self))(n)
似乎一模一样,仅仅是magic
跟self
名字不同。嗯?前者不就是把
magic
自身作为参数传递进自身的返回函数吗?magic(magic)
是把自己传参进去,那么self === magic
。原来
self(self)
自调用的函数,就是magic
自身。于是,我们得到:
看到最终的产物,让人惊呆了。这是什么黑魔法?
仔细一看,原来它就是 lambda 演算的 JavaScript 实现
它不仅适用于阶乘函数的递归处理,任意递归工厂函数经过
Y
函数后,都能得到真正的递归函数。尾声
在这篇文章中,我们有意识地用到的特性只有一个:
我们利用它,让一个函数自己调用自己,然后不断美化美化、简化简化,竟然就构造出了
Y Combinator
。然而:
函数也是值,可传参
中,反推出Y Combinator
,不代表你有多厉害函数也是值,可传参
的定律,却不知道背后的原理就是λ演算
The text was updated successfully, but these errors were encountered: