函数柯里化的应用

函数柯里化,就是将一个接受多个参数的函数转化为接受单一参数的函数的技术。

function curry (func) {
  const args = Array.prototype.slice.call(arguments, 1)

  return function () {
    const args2 = Array.prototype.slice.call(arguments)
    func.apply(this, args.concat(args2))
  }
}

function add (x, y) {
  return x + y
}

var add3 = curry(add, 3)
add3(2) // 5

以上代码中,add3 就是经过柯里化之后的函数。add3 可以理解成是 add 函数的柯里化版本,当调用 add3(2) 时,相当于调用 add(3, 2)

function curry (func) {
  // func.length 返回 func 函数定义中形式参数的数量
  var arity = func.length

  return function () {
    var args = Array.prototype.slice.call(arguments)
    if (args.length >= arity) {
      // 实参数量满足形参的情况
      return func.apply(null, args)
    } else {
      // 实参数量少于形参的情况
      return function () {
        var args2 = Array.prototype.slice.call(arguments)
        return func.apply(null, args.concat(args2))
      }
    }
  }
}

function add (x, y) {
  return x + y
}

var addCurry = curry(add)
addCurry(3, 2) // 5

const add3 = addCurry(3)
add3(2) // 5

以上代码中,调用 curry 函数会返回一个函数,例如 addCurry。当我们向 addCurry 函数传入两个参数时,addCurryadd 没有区别,但是当我们向 addCurry 函数传入一个参数时,addCurryadd 有区别,这时的 addCurry 相当于构造函数,它可以为我们构造出 add 函数的柯里化版本,例如 add3

const match = curry(function (what, x) {
  return x.match(what);
})

const matchEmail = match(emailRegExp)
const matchPhone = match(phoneRegExp)

matchEmail('hello@google.com')
matchPhone('15022222222')

以上代码中,我们利用 match 函数可以构造出各种不同用途的函数。

const filter = curry(function(f, xs) {
  return xs.filter(f);
})

const filterEmail = filter(item => matchEmail(item))
const filterPhone = filter(item => matchPhone(item))

const arr = ['hello@google.com', '15500000000', 'hello@qq.com', '13022211111', 'hello world']

filterEmail(arr) // ['hello@google.com', 'hello@qq.com']
filterPhone(arr) // ['15500000000', '13022211111']

以上代码中,我们利用 filter 函数可以构造出各种不同用途的过滤函数。以下是更多的例子,可以查看源代码

add = curry(function (x, y) {
  return x + y
})

match = curry(function (what, x) {
  return x.match(what)
})

replace = curry(function (what, replacement, x) {
  return x.replace(what, replacement)
})

filter = curry(function (f, xs) {
  return xs.filter(f)
})

map = curry(function map (f, xs) {
  return xs.map(f)
})

reduce = curry(function (f, a, xs) {
  return xs.reduce(f, a)
})

split = curry(function (what, x) {
  return x.split(what)
})

join = curry(function (what, x) {
  return x.join(what)
})

在这里我们可以看到,所有的数据参数都作为了最后一个参数,很明显这样在使用时的好处就在于,这可以成为一种预加载函数函数,非常明显的在于这样柯里化处理函数,可以让这些函数成为底层部署的函数。

参考:一段柯里化函数代码阅读

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 函数和对象 1、函数 1.1 函数概述 函数对于任何一门语言来说都是核心的概念。通过函数可以封装任意多条语句,而且...
    道无虚阅读 10,094评论 0 5
  • 柯里化是函数的一个高级应用,想要理解它并不简单。因此我一直在思考应该如何更加表达才能让大家理解起来更加容易。 通过...
    这波能反杀阅读 69,282评论 117 319
  • 真的是很讨厌他和他妈 人生不太完美的瑕疵点 唉,身体越来越发虚,得认真运动认真读书认真谈尤克里里 莫名觉得被绿 妈...
    小宝尼阅读 1,046评论 0 0
  • 1. 小伊和我,两人有13年的交情。她刚毕业工作,我呢生完孩子,来到新的单位。一直都当姐妹处,无论是生活还是工作中...
    侠女柔情阅读 3,021评论 0 0
  • 下雪了? 下雪了! 下雪了! 一夜之间,大地万物披上了银装!呵,老天爷怕万物生灵承受不了酷冬的寒冷,送羽绒被子来了...
    江雪翁阅读 1,053评论 0 1