Jamey's Jamey's
首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于

Jamey

首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于
  • 深入系列

  • 专题系列

  • underscore系列

    • underscore 系列之如何写自己的 underscore
    • underscore 系列之链式调用
    • underscore 系列之内部函数 cb 和 optimizeCb
    • underscore 系列之内部函数 restArgs
      • 一. partial
      • 二. rest parameter
      • 三. restArgs
      • 四. 优化
      • 五. 性能优化
      • 六. restArgs 与 partial
    • underscore 系列之防冲突与 Utility Functions
    • underscore 系列之实现一个模板引擎(上)
    • underscore 系列之实现一个模板引擎(下)
    • underscore 系列之字符实体与 _.escape
    • underscore 的源码该如何阅读?
  • ES6系列

  • 模块化

  • 正则表达式

  • 单元测试

  • 微前端

  • 实用函数

  • Rollup

  • 解决方案

  • 《JavaScript》笔记
  • underscore系列
Jamey
2021-11-12
目录

underscore 系列之内部函数 restArgs

# underscore 系列之内部函数 restArgs

# 一. partial

在 14.《JavaScript 专题之偏函数》 中,我们写了一个 partial 函数,用来固定函数的部分参数,实现代码如下:

// 这是文章中的第一版
function partial(fn) {
  var args = [].slice.call(arguments, 1);
  return function() {
    var newArgs = args.concat([].slice.call(arguments));
    return fn.apply(this, newArgs);
  };
};
1
2
3
4
5
6
7
8

# 二. rest parameter

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

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

fn(1, 2, 3, 4, 5);
1
2
3
4
5

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

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

写个 demo,测试一下:

function add(a, b) {
  return a + b;
}

var addOne = partial(add, 1);

console.log(addOne(2)); // 3
1
2
3
4
5
6
7

# 三. restArgs

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

我们可以写一个 restArgs 函数,传入一个函数,使用函数的最后一个参数储存剩下的函数参数,使用效果如下:

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

func(1, 2, 3, 4, 5);
1
2
3
4
5

我们来写一版:

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

# 四. 优化

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

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

var func = restArgs(function(a, b, c, d){
  console.log(c); // 报错
})

func(1, 2);
1
2
3
4
5

所以我们再写一版:

// 第二版
function restArgs(func, startIndex) {
  startIndex = startIndex == null ? func.length - 1 : +startIndex;
  return function(){
    var length = Math.max(arguments.length - startIndex, 0);
    var rest = Array(length)
    var index = 0;
    for (; index < length; index++) {
      rest[index] = arguments[index + startIndex]
    }
  
    var args = Array(startIndex + 1);
    for (index = 0; index < startIndex; index++) {
      args[index] = arguments[index]
    }
  
    args[startIndex] = rest;
    return func.apply(this, args)
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

# 五. 性能优化

如果是正常写业务,可能写到这里就结束了,然而 underscore 考虑的更多,鉴于 call 的性能要高于 apply,所以 underscore 做了一个优化:

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

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

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

# 六. restArgs 与 partial

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

var partial = restArgs(function(fn, args){
  return restArgs(function(partialArgs) {
    var newArgs = args.concat(partialArgs);
    return fn.apply(this, newArgs);
  })
})

function add(a, b, c) {
  return a + b + c;
}

var addOne = partial(add, 1);
console.log(addOne(2, 3)); // 6
1
2
3
4
5
6
7
8
9
10
11
12
13
#JavaScript underscore
上次更新: 2022/07/01, 17:34:19
underscore 系列之内部函数 cb 和 optimizeCb
underscore 系列之防冲突与 Utility Functions

← underscore 系列之内部函数 cb 和 optimizeCb underscore 系列之防冲突与 Utility Functions→

Theme by Vdoing | Copyright © 2017-2023 Jamey | blog 闽ICP备19022664号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式