# call、apply、bind

# 手写

# call

Function.prototype.call = function() {
  const args = Array.slice.call(arguments);
  const context = args.shift();
  const temp = Symbol();
  context[temp] = this;
  context[temp](...args);
  delete context[temp];
};
1
2
3
4
5
6
7
8

# apply

Function.prototype.apply = function() {
  const args = Array.slice.call(arguments);
  const context = args.shift();
  const temp = Symbol();
  context[temp] = this;
  context[temp](...args.shift());
  delete context[temp];
};
1
2
3
4
5
6
7
8

# bind(来源于MSN (opens new window))

if (!Function.prototype.bind) {
  Function.prototype.bind = function(oThis) {
    if (typeof this !== "function") {
      // closest thing possible to the ECMAScript 5
      // internal IsCallable function
      throw new TypeError(
        "Function.prototype.bind - what is trying to be bound is not callable"
      );
    }

    var aArgs = Array.prototype.slice.call(arguments, 1),
      fToBind = this,
      fNOP = function() {},
      fBound = function() {
        // this instanceof fBound === true时,说明返回的fBound被当做new的构造函数调用
        return fToBind.apply(
          this instanceof fBound ? this : oThis,
          // 获取调用时(fBound)的传参.bind 返回的函数入参往往是这么传递的
          aArgs.concat(Array.prototype.slice.call(arguments))
        );
      };

    // 维护原型关系
    if (this.prototype) {
      // 当执行Function.prototype.bind()时, this为Function.prototype
      // this.prototype(即Function.prototype.prototype)为undefined
      fNOP.prototype = this.prototype;
    }
    // 下行的代码使fBound.prototype是fNOP的实例,因此
    // 返回的fBound若作为new的构造函数,new生成的新对象作为this传入fBound,新对象的__proto__就是fNOP的实例
    fBound.prototype = new fNOP();

    return fBound;
  };
}
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
32
33
34
35
上次更新时间: 9/12/2021, 10:30:35 PM