lodash_curry函数柯里化的实现

2020-09-19
lodash_curry函数柯里化的实现
 
拉勾大前端学习中 学到柯里化函数 略微有点懵 自己捋了一下其中的逻辑
下面是实现和思路
function curry(fn) { // 形参是fn应该有的参数 在这里就是getSum中的(a,b,c)3个
  // 此处返回的是函数声明而非匿名函数 为的是在传参不足的时候 递归调用自己
  return function curriedFn(...args1) { // 实参是调用时候真正传的参数 有可能是1个 2个或者3个
    console.log(args1);
    // 判断实参和形参的个数
    if (args1.length < fn.length) { // 如果参数不够 返回一个匿名函数出去 当这个匿名函数再次调用的时候 
      // 会将再次调用传的参数和上次调用传的参数合并后再调用函数声明 
      return function (...args2) {
        // 由于这里的args1是匿名函数外的数据 形成了一个闭包 所以匿名函数在执行时args1都是独立的 不会互相影响
        console.log(args1);
        console.log(args2);
        return curriedFn(...args1, ...args2); // 将此次和上次的参数合并再次递归 curriedFn 结果作为返回值
        // 如果args1和args2的长度已经>=形参数 显然再次递归的时候 返回值是 fn(合并后的参数) 也就是fn执行结果作为返回值
        // 如果args1和args2的长度还是<形参的长度 就再返回一个匿名函数作为返回值等待更多的参数传入
      }
    }
    return fn(...args1);
  }
}

function getSum(a, b, c) {
  return a + b + c;
}

let curried = curry(getSum);

b = curried(1); // 执行完curried(1) 打印第4行的 args1 = [1]  返回的是匿名函数 function (...args2) {...}

c = b(2); // 再执行b 也就是function (...args2) {...} 
// 先后打印 10 11行的 args1 = [1] args2 = [2] 然后合并执行curriedFn 打印第三行的args1:[1, 2]
// if 判断 还不满足 再将匿名函数当作此次curriedFn的返回值 在第9行 返回给c

let sum1 = c(3); // 执行c时, 打印第10行的 args1 = [1, 2] 打印第11行的 args2 = [3]
// 组合后再调用 curriedFn 打印第4行的args1 = [1,2,3] if判断已经不满足
// 执行fn(...[1,2,3])作为此次curriedFn的结果 在第9行ruturn出去 也就是最后的结果 

d = b(5); // 每次执行都是一个闭包 其中的args1 都是互不影响的
let sum2 = d(4);
console.log(sum1); // 6 1+2+3
console.log(sum2); // 10 1+5+4

 

打印结果
 
自己捋一遍 思路清晰多了 如果有错 望指出~
 
posted @ 2020-09-19 17:04  蓝小胖纸  阅读(565)  评论(0编辑  收藏  举报