要求写一个函数add(),分别实现能以下效果:html
(1)console.log(add(1)(2)(3)(4)());//10 (2)console.log(add(1,2)(3,4)());//10 (3)console.log(add(1,2)(3,4));//10
针对(1)和(2),有两种思路实现:纯闭包思路和函数柯里化思路。数组
(1)的解决方案(闭包实现)闭包
function add(arg) { // body... let sum = 0; sum+=arg; return function (tmarg) { // body... if (arguments.length == 0) { return sum; }else{ sum+=tmarg; return arguments.callee; } } }
(2)的解决方案app
function add(arg) { // body... let sum = 0; sum = Array.prototype.slice.call(arguments).reduce((a,b) => {return a+b;},sum); return function (tmarg) { // body... if (arguments.length == 0) { return sum; }else{ sum = Array.prototype.slice.call(arguments).reduce((a,b) => {return a+b;},sum); return arguments.callee; } } }
通俗的理解,因为函数柯里化具备这样的特性:它可以"积累"函数的参数(无论是foo(1,2,3)仍是foo(1)(2)(3)这种链式形式)),而且延迟执行。能够将多个参数积累到一个数组中,在最后一步执行求和。
柯里化通用形式:函数
function curry(fn) { // body... var args = Array.prototype.slice.call(arguments,1); return function () { // body... var innerArgs = Array.prototype.slice.call(arguments); var finalArgs = args.concat(innerArgs); console.log(finalArgs); return fn.apply(null,finalArgs); }; }
(2)的解决方案:spa
function add() { let sum = 0; var _args = Array.prototype.slice.call(arguments); var tmpf = function(){ if(arguments.length === 0) { sum = _args.reduce((a,b) => {return a + b;},sum); } _args.push.apply(_args,[].slice.call(arguments)); return tmpf; } }
针对问题(3):prototype
function add(arg) { // body... let sum = 0; sum = Array.prototype.slice.call(arguments).reduce((a,b) => {return a+b;},sum); var tmpf = function (tmarg) { // body... if (arguments.length == 0) { return sum; }else{ sum = Array.prototype.slice.call(arguments).reduce((a,b) => {return a+b;},sum); return tmpf; } }; tmpf.toString = tmpf.valueOf = function () { // body... return sum; } return tmpf; }
完!code
转载请注明原文出处:http://www.cnblogs.com/qcblog/p/6858947.htmlhtm