函数柯里化

详解JS函数柯里化 - 简书

javascript - 「前端面试题系列6」理解函数的柯里化 - 超哥前端小栈 - SegmentFault 思否

维基百科上说道:柯里化,英语:Currying,是把接受多个参数的函数变换成接受一个单一参数的函数,并且返回接受余下的参数而且返回结果的新函数的技术。 

也就是,函数套函数,返回一个函数,剩余参数放子函数

// 普通的add函数
function add(x, y) {
    return x + y
}

// Currying后
function curryingAdd(x) {
    return function (y) {
        return x + y
    }
}

add(1, 2)           // 3
curryingAdd(1)(2)   // 3

柯里化的作用就是,参数复用,延迟执行。

Function.prototype.bind = function (context) {
    var _this = this
    var args = Array.prototype.slice.call(arguments, 1)
 
    return function() {
        return _this.apply(context, args)
    }
}

js中经常使用的bind,实现的机制就是Currying 

实现:

彻底搞懂闭包,柯里化,手写代码,金九银十不再丢分! - 掘金

一、固定长度参数的柯里化

function curry(fn) {
  // 获取原函数的参数长度
  const argLen = fn.length;
  // 跳过序号0,从序号为1开始截取参数,转成数组,保存下来
  const presetArgs = [].slice.call(arguments, 1)
  // 返回一个新函数
  return function() {
    // 新函数调用时会继续传参,整个转成数组保存
    const restArgs = [].slice.call(arguments)
    const allArgs = [...presetArgs, ...restArgs]
    if (allArgs.length >= argLen) {
      // 如果参数够了,就执行原函数
      return fn.apply(this, allArgs)
    } else {
      // 否则继续柯里化
      return curry.call(null, fn, ...allArgs)
    }
  }
}

let m = curry(add)
let v=m(1,1)(1)
console.log(v)

二、动态参数长度的柯里化(推荐)

柯里化与反柯里化 - 掘金

我怎么知道最后执行的时机?其实,这里有个忍者技艺:valueOftoString
js在获取当前变量值的时候,会根据语境,隐式调用valueOftoString方法进行获取需要的值。


function curry(fn) {
  // 跳过序号0,从序号为1开始截取参数,转成数组,保存下来
  const presetArgs = [].slice.call(arguments, 1)
  // 返回一个新函数
  function curried () {
    // 新函数调用时会继续传参
    const restArgs = [].slice.call(arguments)
    const allArgs = [...presetArgs, ...restArgs]
    return curry.call(null, fn, ...allArgs)
  }
  // 重写toString
  curried.toString = function() {
    return fn.apply(null, presetArgs)
  }

  //数值类型
  curried.valueOf = function(){
        return fn.apply(null, presetArgs);
  }

  return curried;
}

function dynamicAdd() {
  const args = [].slice.call(arguments)
  return args.reduce((prev, curr) => {
    return prev + curr
  }, 0)
}
var add = curry(dynamicAdd);


let y1 = add(1)(2)(3)(4) // 10
let y2 = add(1, 2)(3, 4)(5, 6) // 21
console.log(y1.toString())
console.log(y2.toString())

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值