Skip to content

JS 的 call、bind、apply 笔记

Published: at 05:18 PMSuggest Changes

很奇怪的是项目里面很少用到 call,bind,apply 这些,感觉有必要 mark 一下,以后就不用重复查资料了

区别

三者都是用于改变函数体内 this 的指向,但是 bind 与 apply 和 call 的最大的区别是:bind 不会立即调用,而是返回一个新函数,称为绑定函数,其内的 this 指向为创建它时传入 bind 的第一个参数,而传入 bind 的第二个及以后的参数作为原函数的参数来调用原函数。

var obj = {};

function test() {
  console.log(this === obj);
}

test(); //false

var testObj = test.bind(obj);
testObj(); //true

apply 和 call 都是为了改变某个函数运行时的上下文而存在的(就是为了改变函数内部 this 的指向);apply 和 call 的调用返回函数执行结果;

如果使用 apply 或 call 方法,那么 this 指向他们的第一个参数,apply 的第二个参数是一个参数数组,call 的第二个及其以后的参数都是数组里面的元素,就是说要全部列举出来;

bind 语法

func.bind(thisArg[, arg1[, arg2[, ...]]]) thisArg 当绑定函数被调用时,该参数会作为原函数运行时的 this 指向。当使用 new 操作符调用绑定函数时,该参数无效。 arg1, arg2, … 当绑定函数被调用时,这些参数将置于实参之前传递给被绑定的方法。

call 语法

fun.call(thisArg, arg1, arg2, ...) thisArg 在 fun 函数运行时指定的 this 值。需要注意的是,指定的 this 值并不一定是该函数执行时真正的 this 值,如果这个函数处于非严格模式下,则指定为 null 和 undefined 的 this 值会自动指向全局对象 (浏览器中就是 window 对象),同时值为原始值 (数字,字符串,布尔值) 的 this 会指向该原始值的自动包装对象。 arg1, arg2, … 指定的参数列表。

apply 语法

fun.apply(thisArg, [argsArray]) thisArg 在 fun 函数运行时指定的 this 值。需要注意的是,指定的 this 值并不一定是该函数执行时真正的 this 值,如果这个函数处于非严格模式下,则指定为 null 或 undefined 时会自动指向全局对象(浏览器中就是 window 对象),同时值为原始值(数字,字符串,布尔值)的 this 会指向该原始值的自动包装对象。 argsArray: 一个数组或者类数组对象,其中的数组元素将作为单独的参数传给 fun 函数。如果该参数的值为 null 或 undefined,则表示不需要传入任何参数。从 ECMAScript 5 开始可以使用类数组对象。

区别总结

当我们使用一个函数需要改变 this 指向的时候才会用到 call,apply,bind 如果你要传递的参数不多,则可以使用 fn.call(thisObj, arg1, arg2 …) 如果你要传递的参数很多,则可以用数组将参数整理好调用 fn.apply(thisObj, [arg1, arg2 …]) 如果你想生成一个新的函数长期绑定某个函数给某个对象使用,则可以使用

const newFn = fn.bind(thisObj);
newFn(arg1, arg2...)

call

call() 方法使用一个指定的 this 值和单独给出的一个或多个参数来调用一个函数。

注意:如果 call 方法没有参数,或者参数为 null 或 undefined,则等同于指向全局对象。

window.color = "red";
var o = { color: "blue" };
function sayColor() {
  alert(this.color);
}
sayColor.call(this); //red
sayColor.call(window); //red
sayColor.call();
sayColor.call(null);
sayColor.call(undefined);
sayColor.call(o); //blue

判断对象类型

var arr = [];
Object.prototype.toString.call(arr); // [object Array]
//把函数体 Object.prototype.toString() 方法内部的 this,绑到 arr 的执行环境(作用域)

各种手写 call

var foo = {
  count: 1,
};
function bar() {
  console.log(this.count);
}
bar.myCall(foo); // 1
Function.prototype.myCall = function (context) {
  // 取得传入的对象(执行上下文),比如上文的 foo 对象,这里的 context 就相当于上文的 foo
  // 不传第一个参数,默认是 window,
  var context = context || window;
  // 给 context 添加一个属性,这时的 this 指向调用 myCall 的函数,比如上文的 bar 函数
  context.fn = this; //这里的 context.fn 就相当于上文的 bar 函数
  // 通过展开运算符和解构赋值取出 context 后面的参数,上文的例子没有传入参数列表
  var args = [...arguments].slice(1);
  // 执行函数(相当于上文的 bar(...args))
  var result = context.fn(...args);
  // 删除函数
  delete context.fn;
  return result;
};
/**
 * 每个函数都可以调用 call 方法,来改变当前这个函数执行的 this 关键字,并且支持传入参数
 */
Function.prototype.myCall = function (context) {
  //第一个参数为调用 call 方法的函数中的 this 指向
  var context = context || global;
  //将 this 赋给 context 的 fn 属性
  context.fn = this; //此处 this 是指调用 myCall 的 function

  var arr = [];
  for (var i = 0, len = arguments.length; i < len; i++) {
    arr.push("arguments[" + i + "]");
  }
  //执行这个函数,并返回结果
  var result = eval("context.fn(" + arr.toString() + ")");
  //将 this 指向销毁
  delete context.fn;
  return result;
};

apply

apply() 方法调用一个具有给定 this 值的函数,以及以一个数组(或类数组对象)的形式提供的参数。

示例

var foo = {
  count: 1,
};
function bar() {
  console.log(this.count);
}
bar.myApply(foo); // 1

应用场景找出数组中最大或最小的元素

var a = [10, 2, 4, 15, 9];
Math.max.apply(Math, a); //15
Math.min.apply(null, a); //2

可以将一个类似(伪)数组的对象(比如 arguments 对象)转为真正的数组。**前提:**被处理的对象必须有 length 属性,以及相对应的数字键。

//接收的是对象,返回的是数组
Array.prototype.slice.apply({ 0: 1, length: 1 }); // [1]
Array.prototype.slice.apply({ 0: 1 }); // []
Array.prototype.slice.apply({ 0: 1, length: 2 }); // [1, undefined]
Array.prototype.slice.apply({ length: 1 }); // [undefined]
//(切下)[].slice(1, n),返回索引为 1 到索引为 n-1 的数组

数组追加

var arr1 = [1, 2, 3];
var arr2 = [4, 5, 6];
[].push.apply(arr1, arr2);
console.log(arr1); //[1, 2, 3, 4, 5, 6]
console.log(arr2); //[4, 5, 6]

数组合并

var arr1 = [1, 2, { id: 1, id: 2 }, [1, 2]];
var arr2 = ["ds", 1, 9, { name: "jack" }];
// var arr = arr1.concat(arr2);//简单做法
Array.prototype.push.apply(arr1, arr2);
console.log(arr1);

各种手写 apply

Function.prototype.myApply = function (context) {
  var context = context || window;
  context.fn = this;
  var result;
  // 判断第二个参数是否存在,也就是 context 后面有没有一个数组
  // 如果存在,则需要展开第二个参数
  if (arguments[1]) {
    result = context.fn(...arguments[1]);
  } else {
    result = context.fn();
  }
  delete context.fn;
  return result;
};
/**
 * apply 函数传入的是 this 指向和参数数组
 */
Function.prototype.myApply = function (context, arr) {
  var context = context || global;
  context.fn = this;
  var result;
  if (!arr) {
    result = context.fn(); //直接执行
  } else {
    var args = [];
    for (var i = 0, len = arr.length; i < len; i++) {
      args.push("arr[" + i + "]");
    }
    result = eval("context.fn([" + args.toString() + "])");
  }
  //将 this 指向销毁
  delete context.fn;
  return result;
};

bind

bind() 方法创建一个新的函数,在 bind() 被调用时,这个新函数的 this 被指定为 bind() 的第一个参数,而其余参数将作为新函数的参数,供调用时使用。

示例

var a = {
  name: "Cherry",
  fn: function (a, b) {
    console.log(a + b);
  },
};
var b = a.fn;
b.call(a, 1, 2); //立即调用该函数
b.bind(a, 1, 2)(); //手动调用 (),它返回一个原函数的拷贝(新的,不是原函数),并拥有指定的 this 值和初始参数。

各种手写 bind

Function.prototype.myBind = function () {
  var _this = this;
  var context = [].shift.call(arguments); // 保存需要绑定的 this 上下文
  var args = [].slice.call(arguments); //剩下参数转为数组
  console.log(_this, context, args);
  return function () {
    return _this.apply(context, [].concat.call(args, [].slice.call(arguments)));
  };
};
Function.prototype.myBind = function (context) {
  if (typeof this !== "function") {
    throw new TypeError("Error");
  }
  var _this = this;
  var args = [...arguments].slice(1);
  // 返回一个函数
  return function F() {
    // 因为返回了一个函数,我们可以 new F(),所以需要判断
    if (this instanceof F) {
      return new _this(...args, ...arguments);
    }
    return _this.apply(context, args.concat(...arguments));
  };
};

Previous Post
解决 Axios 额外发起一次 OPTIONS 请求
Next Post
使用 JavaScript 加载其他 JS 文件