柯里化面试题及方法
// 实现一个add方法,使计算结果能够满足如下预期:
add(1)(2)(3) = 6;
add(1, 2, 3)(4) = 10;
add(1)(2)(3)(4)(5) = 15;
function add() {
// 第一次执行时,定义一个数组专门用来存储所有的参数
var _args = Array.prototype.slice.call(arguments);
// 在内部声明一个函数,利用闭包的特性保存_args并收集所有的参数值
var _adder = function() {
_args.push(...arguments);
return _adder;
};
// 利用toString隐式转换的特性,当最后执行时隐式转换,并计算最终的值返回
_adder.toString = function () {
return _args.reduce(function (a, b) {
return a + b;
});
}
return _adder;
}
利用toString隐式转换的特性
上面的 add(1)(2)(3) = 6 这一句其实误导了大家,让人以为add(1)(2)(3)的执行结果是返回一个6。
// 当我们打印执行结果时,并不是6
console.log(add(1,2)(3)); // { [Function: _adder] toString: [Function] }
// 只有这样打印才会输出6
console.log(add(1,2)(3).toString()); // 6
那么那里应用了toString隐式转换的特性呢,当作为判断条件或者出现, 或者作为运算符的某一项时,toString会隐式执行。
// 作为判断条件时,toString会隐式执行
console.log(add(1,2)(3) == 6); // true toString悄悄执行了,将收集到的参数进行累加
// 或者作为运算符的某一项时,toString也会隐式执行
console.log(add(1,2)(3)+1); // 7
console.log(add(1,2)(3)-1); // 5
console.log(add(1,2)(3)*1); // 6
console.log(add(1,2)(3)/2); // 3
这样看来add(1,2)(3)与6几乎没有区别,无论是拿去进行判断还是计算,它都是与6对等的。
但是注意当进行判断时,如果使用的是全等于add(1,2)(3) === 6,则会被判断为false。