什么是函数柯里化
在计算机科学中,柯里化(Currying)是把接受多个参数的函数变换成接受一个单一参数(最初函数的第一个参数)的函数,并且返回接受余下的参数且返回结果的新函数的技术。这个技术由 Christopher Strachey 以逻辑学家 Haskell Curry 命名的,尽管它是 Moses Schnfinkel 和 Gottlob Frege 发明的。
举个例子,假设程序员A是个妻管严,工资悉数上交给妻子,为了买一个心爱的键盘
程序员A每天都偷偷藏几毛钱,打算年底买个键盘,因此程序员A写了以下这个方法用来统计年终一共凑了多少钱,方法如下:
/** * @description 统计金额 * @return {number} */ function countMoney() { let money = 0 // 温馨提示:arguments是所接收的所有参数组成的类数组,不懂的需要搜一搜补补知识啦 for (let i = 0; i < arguments.length; i++) { money += arguments[i] } return money } // 藏了一年的账本记录的数据 const records = [1, 1, 2, 2, 3, 3, 4, 4] // 把全部数据都输入进行计算 countMoney(1, 1, 2, 2, 3, 3, 4, 4)
上面代码所呈现的方法,是最直接的计算方法,其不方便的地方在于,程序员A还要拿个小本本把每天存了多少钱先记录下来!!!万一这个本子被发现那就少不了跪键盘了,极度不安全。
更安全地藏私房钱
实际上,每天记录下当前的数据是不灵活的,而函数柯里化则有效地解决了这个问题。
我们想要这样存储我们的私房钱
// 2018-01-01 存了1毛钱 countMoney(1) // 2018-01-02 存了2毛钱 countMoney(2) // 2018-01-03 存了3毛钱 countMoney(3) // 2018-01-04 存了4毛钱 countMoney(4) //一年以后 // 统计这笔巨额存款 countMoney()
上述的这种方法中,我们不关心数据的存储记录,我们只需要每天往存钱罐里面塞钱,然后年底取出来就是一个总和。再也不担心留下小本本作为证据了!
函数柯里化代码的实现
/** * @description countMoney为立即执行函数,返回的结果是另一个函数 */ const countMoney = (function () { let money = 0 let args = [] const res = function () { if (arguments.length === 0) { for (let i = 0; i < args.length; i++) { money += args[i] } return money } else { // arguments 是个类数组来着,应该用展开符展开才能push进去 args.push(...arguments) return res } } return res })() // 2018-01-01 存了1毛钱 countMoney(1) // 2018-01-02 存了2毛钱 countMoney(2) // 2018-01-03 存了3毛钱 countMoney(3) // 2018-01-04 存了4毛钱 countMoney(4) //一年以后 // 统计这笔巨额存款 输出结果为 10 console.log(countMoney()) // 你还可以装逼地进行花式统计,结果同样是10 countMoney(1)(2)(3)(4)()
分析代码
实际上,在JavaScript的很多思想和设计模式中,闭包是个很常见且很重要的东西,上述的代码中,本质上就是利用了闭包。
该函数是个立即执行函数,返回了一个新函数,而这个新函数实际上就是一个闭包,这个新函数把每次接收到的参数都存储起来,
并且继续返回一个新函数,当发现某次调用的时候没有传入参数,那就意味着要进行数据统计,从而把之前存储的数据一次性拿
出来计算,最后返回计算结果。其流程如下:
总结
所谓的函数柯里化,亦或者在开发中涉及到的其他一些概念,例如闭包、单例模式、观察者模式等等都好,我们需要关注的点在于掌握
这些模式或者概念中的代码设计思想,从而更好地服务于我们的业务开发,让我们的代码更健壮、灵活、高效。