rest parameter

ES6 为我们提供了剩余参数(rest parameter)语法,允许我们将一个不定数量的参数表示为一个数组。

  1. function fn(a, b, ...args) {
  2. console.log(args); // [3, 4, 5]
  3. }
  4.  
  5. fn(1, 2, 3, 4, 5)

我们可以利用这一特性简化 partial 实现的代码:

  1. function partial(fn, ...args) {
  2. return function(...partialArgs) {
  3. var newArgs = args.concat(partialArgs);
  4. return fn.apply(this, newArgs);
  5. };
  6. };

写个 demo,测试一下:

如果不使用 … 拓展操作符,仅用 ES5 的内容,该怎么实现呢?

  1. var func = restArgs(function(a, b, c){
  2. console.log(c); // [3, 4, 5]
  3. })
  4.  
  5. func(1, 2, 3, 4, 5)

我们来写一版:

  1. // 第一版
  2. function restArgs(func) {
  3. return function(){
  4. // startIndex 表示使用哪个位置的参数用于储存剩余的参数
  5. var startIndex = func.length - 1;
  6. var length = arguments.length - startIndex;
  7.  
  8. var rest = Array(length)
  9. var index = 0;
  10.  
  11. // 使用一个数组储存剩余的参数
  12. // rest [3, 4, 5]
  13. for (; index < length; index++) {
  14. rest[index] = arguments[index + startIndex]
  15. }
  16.  
  17. // args [1, 2, undefined]
  18. var args = Array(startIndex + 1);
  19. for (index = 0; index < startIndex; index++) {
  20. args[index] = arguments[index]
  21. }
  22.  
  23. // args [1, 2, [3, 4, 5]]
  24. args[startIndex] = rest;
  25.  
  26. return func.apply(this, args)
  27. }
  28. }

优化

我们默认使用传入的函数的最后一个参数储存剩余的参数,为了更加灵活,我们可以再增加一个参数,用来指定 startIndex,如果没有指定,就默认使用最后一个参数。

此外,注意,我们使用 Array(length) 创建数组,而 length 的计算方式是 arguments.length - startIndex,这个值有可能是负数!比如:

所以我们再写一版:

  1. // 第二版
  2. function restArgs(func, startIndex) {
  3. startIndex = startIndex == null ? func.length - 1 : +startIndex;
  4. return function(){
  5. var length = Math.max(arguments.length - startIndex, 0);
  6. var rest = Array(length)
  7. var index = 0;
  8. for (; index < length; index++) {
  9. rest[index] = arguments[index + startIndex]
  10. }
  11.  
  12. var args = Array(startIndex + 1);
  13. for (index = 0; index < startIndex; index++) {
  14. args[index] = arguments[index]
  15. }
  16.  
  17. return func.apply(this, args)
  18. }
  19. }
  1. // 第三版
  2. var restArgs = function(func, startIndex) {
  3. startIndex = startIndex == null ? func.length - 1 : +startIndex;
  4. return function() {
  5. var length = Math.max(arguments.length - startIndex, 0),
  6. rest = Array(length),
  7. index = 0;
  8.  
  9. for (; index < length; index++) {
  10. rest[index] = arguments[index + startIndex];
  11. }
  12.  
  13. // 增加的部分
  14. switch (startIndex) {
  15. case 0:
  16. return func.call(this, rest);
  17. case 1:
  18. return func.call(this, arguments[0], rest);
  19. case 2:
  20. return func.call(this, arguments[0], arguments[1], rest);
  21. }
  22.  
  23. var args = Array(startIndex + 1);
  24. for (index = 0; index < startIndex; index++) {
  25. args[index] = arguments[index];
  26. }
  27.  
  28. args[startIndex] = rest;
  29. return func.apply(this, args);
  30. };

至此,restArgs 函数就完成了,underscore 很多函数比如 invoke、without、union、difference、bind、partial、bindAll、delay 都用到了 restArgs 函数。

当使用 underscore 的时候,我们可以以 _.restArgs 的形式调用该函数。

restArgs 与 partial

最后,使用我们的写的 restArgs 函数重写下 partial 函数:

underscore 系列目录地址:https://github.com/mqyqingfeng/Blog

如果有错误或者不严谨的地方,请务必给予指正,十分感谢。如果喜欢或者有所启发,欢迎 star,对作者也是一种鼓励。