Geass's Studio.

基于 Symbol.toPrimitive 构建 sum 的柯里化函数

字数统计: 673阅读时长: 2 min
2019/04/27 Share

在 js 中,如何编写一个可无限调用的柯里化 sum 函数?(sum(1,2,3)(4)()(5))。在下文中,就让我们一起实现它。而在实现前,我们需要引入两个知识点:柯里化与 Symbol.toPrimitive

柯里化

首先,什么是柯里化?柯里化是函数式编程中的一个重要特性,是把接受多个参数的函数变换成接受部分参数的函数,并且返回接受余下的参数而且返回结果的新函数的技术。

例如: func(a, b, c, d) => func(a)(b)(c)(d) 就是一个函数的柯里化应用。柯里化函数的优点是可以很方便的基于现有的通用函数构建出一定特性的新函数。我们举一个简单的例子

1
2
3
4
5
6
const multiply = x => y => x * y;

// 需求是获取计算6的倍数的函数
const multiply6 = multiply(6)

multiply6(10); // 60

Symbol.toPrimitive

在 js 中,基本数据类型有 string, number,boolean,null,undefined,symbol 这六种。Symbol.toPrimitive 简写为 @@toPrimitive

tc39 规范中,@@toPrimitive 使用 ToPrimitive ( input [ , PreferredType ] ) 抽象操作实现。操作接收一个 input 参数与一个额外的 PreferredType 参数。当一个对象能够转换为多个原始值的时候,将根据 PrefferedType 进行如下算法转换:

  1. 校验 input 值符合 ECMAScript 规范
  2. 如果 type(input) 不为 object 则直接返回
  3. 如果 type(input)object
    1. 如果未定义 PreferredType,则设置 hint 为默认 default
    2. 如果定义 PreferredType 为 hint String,则设置 hintstring
    3. 如果定义 PreferredType 为 hint number, 则设置 hintnumber
    4. 如果 input 有定义 @@toPrimitive 方法
      1. 调用方法 @@toPrimitive(input, hint) 获取结果 result
      2. 如果 type(result) 不为 object, 返回最终结果
      3. 如果 type(result)object,抛出 TypeError 异常
    5. 如果 hintdefault,设置 hintnumber
    6. 调用原生 OrdinaryToPrimitive 方法求值
    7. 如果 hintnumber, 则依照 valueOf, toString 的顺序调用函数
    8. 如果 hintstring, 则依照 toStringvalueOf 的顺序调用函数
    9. 若函数存在且结果类型不为 object,则返回最终结果
    10. 否则抛出 TypeError 异常

通过 @@toPrimitive 方法,我们可以自定义函数的原始值 (ps. js 里函数也为对象

由此,我们可以写出如下 sum 函数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
const sum = (...xs) => {
const theSum = (args) => args.reduce((a, b) => a+b, 0)
const go = (acc) => {
const result = (...ys) => go(theSum([acc, ...ys]));
// 定义 @@toPrimitive 方法
result[Symbol.toPrimitive] = hint =>
hint === 'string' ? string(acc) : acc;
return result
}
return go(theSum(xs));
}

console.log(sum(1, 2, 3) + 1); // 7
console.log(sum(1)(2)(3) + 1); // 7
console.log(sum(1)(2)(3) + '1'); // 61

因为定义了 Symbol.toPrimitive 方法,result 便具有了函数和作为 stringnumber 的多重身份,从而实现了我们的目标。

CATALOG
  1. 1. 柯里化
    1. 1.1. Symbol.toPrimitive