ES11 的自动 Curry 与自动柯里化详解
在前端开发中,函数式编程逐渐成为越来越热门的技术,而柯里化(Currying)作为其中的一种技巧,也逐渐被人们所接受。ES6 的箭头函数已经大大简化了函数的书写方式,但是在柯里化中,却还需要手动传递参数。ES11 中新增了自动 Curry 和自动柯里化的特性,极大地提升了函数的编写效率。本篇文章将介绍 ES11 自动 Curry 和自动柯里化的详细原理和应用。
一、Curry 的概念介绍
Curry 技术是由 Haskell Curry 命名的,它是一种约定,即一个函数在需要多个参数时,先返回一个接受第一个参数的新函数,再由新函数返回一个接受第二个参数的新函数,以此类推,直到接受了所有参数。这个时候,函数被完全调用,返回最终结果。它的具体实现方式如下:
-- -------------------- ---- ------- -------- ------ - ------ ----------- - ------ ----------- - ------ - - - - -- - - - --- ------- - ------------- --- ------- - ------ ------ --------------------- -- - --------------------- -- ---------- ------------- ----------- -- --- - --------
此处定义了一个 add 函数,每次接收一个参数并返回一个函数,最终返回三个参数的和。在使用时,可以采用 add(1)(2)(3) 的方式完成调用,而 add(1, 2)(3) 这种方式是不能通过的,因为 add 函数只接受一个参数,多个参数的时候也要返回一个新的函数。
二、ES11 自动 Curry 的实现方式
在 ES11 中,提供了一个新的方法 '' "curry" '' ,通过这个方法可以让函数自动实现柯里化,从而省略手动传递参数的步骤。实现自动 Curry 的关键在于优化原函数的参数个数,使其能够自动根据需要返回新函数,并形成柯里化的调用链。其实现方式如下:
使用 Proxy 对象检测原函数,当调用原函数时,如果参数数目不足,则返回一个新的 Proxy 对象,等待剩余参数的调用,当传递完所有参数时,返回最终的计算结果。
function add(x, y, z) { return x + y + z; } let curryAdd = add.curry(); console.log(curryAdd(1)(2)(3)); // 6
在 ES11 中,通过定义一个 Proxy 对象来检测函数的调用,从而实现了函数的自动 Curry。上述代码中,我们直接调用了 add 函数,但是通过调用 add.curry() 方法,返回了一个新的函数 curryAdd。当我们通过 curryAdd(1)(2)(3) 的方式调用时,输出的结果和原始的 add 方法是相同的,而这个过程在 curryAdd 内部自动完成了该函数的柯里化。
三、ES11 自动柯里化的使用限制
除了自动 Curry 实现之外,ES11 中还添加了一种新的语法 '' "?." '',可以自动实现柯里化。例如:
let obj = { data: { value: 100 }}; let value = obj?.data?.value;
通过这种方式访问对象属性时,如果在对象上某个属性不存在,则直接返回 undefined。这种语法的使用非常方便,但是需要注意,这种自动柯里化的功能是有限制的。当使用 '' "?." '' 访问对象属性时,只有在对象存在时才会返回值,否则会直接抛出一个 ReferenceError 错误。这意味着,如果需要使用该特性,需要在调用之前先确保这个属性或者方法实际存在。
另外,? 特性支持链式操作,但只在对象属性和方法的调用上生效,如果链式调用中存在函数参数,则需要使用 "curry" 方法手动解决。
-- -------------------- ---- ------- -------- ------ -- -- - ------ - - - - -- - --- --- - - ----- - ------ --- --- --- ----- - ----------------- --- -------- - ------------ ------------------------------------- -- - ------------------------- --------- -- - ------------------------------ ---- -- - ------------------------- -- ---- -- -
在该例中,我们使用了 ? 特性和 curry() 方法来实现自动 Curry 和柯里化,从而大大简化了代码的书写。
结论
ES11 中新增的自动 Curry 和自动柯里化特性,可以大幅简化函数的书写,同时也可以避免需要手动传递参数的问题。虽然在柯里化的过程中仍有一些限制和需要注意的事项,但是仍有很大的帮助作用。因此,我们可以在实际开发中灵活地应用这些特性,从而提升代码的效率和可读性。
来源:JavaScript中文网 ,转载请注明来源 https://www.javascriptcn.com/post/67296e512e7021665e24734a