今天在做到一个题目,题目的描述如下:
已知 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); } }