今天在做到一个题目,题目的描述如下:
已知 fn 为一个预定义函数,实现函数 curryIt,调用之后满足如下条件:
1、返回一个函数 a,a 的 length 属性值为 1(即显式声明 a 接收一个参数)
2、调用 a 之后,返回一个函数 b, b 的 length 属性值为 1
3、调用 b 之后,返回一个函数 c, c 的 length 属性值为 1
4、调用 c 之后,返回的结果与调用 fn 的返回值一致
5、fn 的参数依次为函数 a, b, c 的调用参数
//输入:
var fn = function (a, b, c) {return a + b + c}; curryIt(fn)(1)(2)(3);
//输出:6
我给出的答案:
function curryIt(fn) {
return function a(a){
return function b(b){
return function c(c){
return fn.call(this,a,b,c);
}
}
}
}
后来看到社区的讨论,觉得自己没有抓住题干的重点,最近从刷题做起,在社区看到很多其他人解答同一道题目很开拓的思维,看到新的知识点和新颖的思维方式,就记录下来吧。
柯里化
>
柯里化就是把接收多个参数的函数变换成为接收单一参数(最初函数的第一个参数)的函数,并且返回接受余下参数且返回结果的新函数的技术。
用柯里化却解答这道题:
>
我们将预定义的函数的参数逐一传入curryIt中,当参数完全传入之后,就执行预定义的函数。
所以,我们首先要做的就是获取预定义函数的参数的个数fn.length,然后声明一个空数组存放这些参数。返回一个匿名函数接收参数并执行,当参数个数小于fn.length,则再次返回该匿名函数,继续接收参数并执行,直到参数个数等于fn.length.最后调用apply执行预定的函数。反思我自己使用的方法,直接把fn接收参数的个数写死了,但是柯里化方法是根据预定义的函数的接收参数的个数,来数组中存放的参数的个数。
最终代码:
function curryIt(fn){
//获取预定义函数的参数个数
var n = fn.length;
//声明存放参数的数组
var args = [];
return function(arg){
args.push(arg);
if(args.length < n){
// callee 属性是一个指针,指向拥有这个 arguments 对象的函数
return arguments.callee;
}else
return fn.apply(this,args);
}
}