柯里化函数及占位符的实现

2021-04-03  本文已影响0人  小杰66
//柯里化函数
//fn.length默认为函数形参的个数
function curry(fn, len = fn.length) {
  return _curry.call(this, fn, len);
}

function _curry(fn, len, ...args) {
  //curry后返回一个函数,闭包保存原函数fn,需要的参数个数len,已传入的参数
  //返回的函数执行后 加入新传入的参数 判断是否满足执行条件 满足则运行原函数 否则重复上一步
  return function (...params) {
    var _args = [...args, ...params];
    if (_args.length >= len) {
      return fn.call(this, ..._args);
    } else {
      return _curry.call(this, fn, len, ..._args);
    }
  };
}

function sum(a, b, c, d, e, f, g) {
  console.log(a, b, c, d, e, f, g);
}
curry(sum)(1)(2, 3)(4)(5, 6, 7);

//占位符柯里化
function curry(fn, length = fn.length, holder = curry) {
  return _curry.call(this, fn, length, holder);
}

function _curry(fn, len, holder, args = [], holders = []) {
  return function (...params) {
    var _args = [...args];
    var _holders = [...holders];
    params.forEach(function (p) {
      //当前传入的参数不是占位符
      if (p !== holder) {
        //上一轮还有未替换的占位符 则替换
        if (holders.length) {
          //获取上一轮还未替换的占位符位置
          var index = holders.shift();
          //在本轮记录的占位符中 删除替换的占位符 为什么不直接删除第一个元素而要查找位置呢? 因为有可能上一轮的占位符在本轮中再次被占位
          _holders.splice(_holders.indexOf(index), 1);
          //替换占位符的值
          _args[index] = p;
        } else {
          //否则只要添加当前参数
          _args.push(p);
        }
      } else {
        //当前传入的参数是占位符
        if (holders.length) {
          //还存在上一轮未替换的占位符,则需要再次占用该位置,此时该占用属于本轮占用,从上一轮holders中删除即可 本轮的_holders中仍保留着该占位符
          holders.shift();
        } else {
          //上一轮的占位符已经全部替换时 直接记录占位符
          _args.push(p);
          _holders.push(_args.length - 1);
        }
      }
    });

    if (_args.length >= len && _args.slice(0, len).every((p) => p !== holder)) {
      return fn.call(this, ..._args);
    } else {
      return _curry.call(this, fn, len, holder, _args, _holders);
    }
  };
}

function sum(a, b, c, d, e, f, g) {
  console.log(a, b, c, d, e, f, g);
}
curry(sum)(1)(curry, curry, 4)(curry, 3)(2, 5, 6, 7);
上一篇 下一篇

猜你喜欢

热点阅读