前言

在 JavaScript 中,apply、bind 和 call 是三个重要的函数,它们都是 Function.prototype 的方法。这些函数可以让我们动态地改变函数的 this 值,或者传递参数来执行函数。本篇博客将详细介绍 apply、bind 和 call 的使用方法以及它们之间的区别。

apply

apply() 方法是 Function.prototype 上的一个方法,可以用于改变函数的 this 值。它接受两个参数:第一个参数是要绑定的 this 值,第二个参数是一个数组,其中包含要传递给函数的参数。apply() 方法会立即执行函数。


function addNumbers(a, b, c) {
return a + b + c;
} const numbers = [1, 2, 3]; const result = addNumbers.apply(null, numbers);
console.log(result); // 输出 6

在上面的例子中,我们定义了一个 addNumbers() 函数,它接受三个参数并返回它们的和。然后,我们定义了一个数组 numbers,其中包含要传递给 addNumbers() 函数的参数。我们使用 apply() 方法将数组作为参数传递给函数,并将 this 值设置为 null。最后,我们打印出 addNumbers() 函数的返回值。

call

call() 方法和 apply() 方法非常相似,但是它的参数不是数组,而是一个一个传递的。call() 方法也可以用于改变函数的 this 值。

function greet(name) {
console.log(`Hello, ${name}!`);
} greet.call(null, 'Alice'); // 输出 "Hello, Alice!"

在上面的例子中,我们定义了一个 greet() 函数,它接受一个参数 name,并在控制台中输出一条欢迎信息。我们使用 call() 方法将 this 值设置为 null,并将要传递给函数的参数一个一个地传递。

bind

bind() 方法和 apply() 方法和 call() 方法有些不同。它不会立即执行函数,而是返回一个新的函数,这个函数的 this 值已经被改变了。我们可以在之后的任何时候调用这个函数。并且传递的部分参数也可在后面使用。

const person = {
name: 'Alice',
greet: function() {
console.log(`Hello, my name is ${this.name}!`);
}
}; const greetAlice = person.greet.bind(person);
greetAlice(); // 输出 "Hello, my name is Alice!"

在上面的例子中,我们定义了一个 person 对象,其中包含一个 greet() 方法。然后,我们使用 bind() 方法创建一个新的函数 greetAlice,并将 person 对象作为 this 值绑定。最后,我们调用 greetAlice() 函数,它会输出 "Hello, my name is Alice!"。

apply()、call() 和 bind() 的区别

apply()、call() 和 bind()三个方法之间的最大区别在于它们传递参数的方式以及它们是否是立即执行函数:

  • apply() 和 call() 方法立即执行函数并且传递参数的方式不同。
  • apply() 方法传递一个数组作为参数,而 call() 方法一个个传递参数。
  • bind() 方法不会立即执行函数,而是返回一个新的函数,这个函数的 this 值已经被绑定了。

在使用这些方法时,我们需要理解它们之间的区别,并根据具体情况进行选择。如果我们想要立即执行一个函数并传递一个数组作为参数,那么应该使用 apply() 方法。如果我们想要立即执行一个函数并逐个传递参数,那么应该使用 call() 方法。如果我们想要创建一个新的函数并绑定 this 值,那么应该使用 bind() 方法。

此外,在使用 apply()、call() 和 bind() 方法时,我们需要注意传递的 this 值和参数是否正确。如果我们不传递 this 值或传递一个错误的 this 值,那么函数可能无法正确地执行。同样,如果我们没有正确地传递参数,那么函数的结果也可能会出现错误。

内部机制

了解这些方法的内部机制可以帮助我们更好地理解它们的使用方式。在 JavaScript 中,函数是对象,因此每个函数都有一个 prototype 对象。当我们创建一个函数时,它会自动创建一个 prototype 对象,并将该对象的 constructor 属性设置为函数本身。

apply()、call() 和 bind() 方法都是在 Function.prototype 上定义的(每个函数都是 Function 对象的实例),因此每个函数都可以使用它们。当我们调用 apply() 或 call() 方法时,JavaScript 引擎会将 this 值设置为传递给方法的第一个参数,并将要传递给函数的参数作为数组或单个参数传递。当我们调用 bind() 方法时,它会返回一个新的函数,并将 this 值设置为绑定的值。

下面是 apply() 方法的一个简单实现:

Function.prototype.myApply = function(thisArg, args) {
// 首先判断调用 myApply() 方法的对象是否为函数
if (typeof this !== 'function') {
throw new TypeError('Function.prototype.myApply called on non-function');
} // 如果 thisArg 为 null 或 undefined,则将其设置为全局对象
if (thisArg === null || thisArg === undefined) {
thisArg = window;
} // 将函数设置为 thisArg 的方法,以便在 thisArg 上调用它
thisArg.__myApply__ = this; // 执行函数并获取结果
const result = thisArg.__myApply__(...args); // 删除在 thisArg 上设置的方法
delete thisArg.__myApply__; // 返回函数的执行结果
return result;
};

call() 方法的实现机制与 apply() 方法类似,只是在调用函数时将参数传递给函数的方式有所不同。call() 方法的实现如下:


Function.prototype.myCall = function(thisArg, ...args) {
// ... 如上 只是参数需要用...rest 展开 };

bind() 方法的实现机制稍微有些不同。bind() 方法会创建一个新的函数,并将原函数的 this 值绑定到指定的对象上。当新函数被调用时,它会将绑定的 this 值传递给原函数,并将任何附加的参数一并传递。bind() 方法的实现如下:

Function.prototype.myBind = function(thisArg, ...args) {
// 首先判断调用 myBind() 方法的对象是否为函数
if (typeof this !== 'function') {
throw new TypeError('Function.prototype.myBind called on non-function');
} // 保存原函数的引用
const originalFunc = this; // 返回一个新函数
return function boundFunc(...newArgs) {
// 将 this 值设置为 thisArg,并调用原函数
return originalFunc.apply(thisArg, [...args, ...newArgs]);
};
};

这里,我们通过使用剩余参数和展开语法来处理任意数量的传递参数,同时使用 apply() 方法将绑定的 this 值和所有参数传递给原函数。

需要注意的是,这里的 bind() 方法是定义在 Function 的原型上的。这意味着,所有的函数都可以调用这个方法,并将其绑定到需要的对象上。

值得一提的是,bind() 方法在绑定 this 值的同时,还可以通过将附加参数传递给新函数来“柯里化”函数。也就是说,通过预先绑定部分参数,我们可以创建一个新函数,这个函数接受剩余的参数,并在调用原函数时将这些参数一并传递。这个技巧在实际开发中经常用到,可以帮助我们简化代码,并提高代码的可读性。

以上是 apply()、call() 和 bind() 方法的手动实现方式,当然,JavaScript 中的这些方法已经预先实现好了,我们只需要使用即可

apply()、call() 和 bind() 方法的使用场景

1. 改变函数的 this 值

在 JavaScript 中,函数的 this 值默认指向全局对象(在浏览器中通常为 window 对象)。如果我们想要在函数内部使用不同的 this 值,那么可以使用 apply()、call() 或 bind() 方法来改变它。例如:

const obj = { name: 'Alice' };

function greet() {
console.log(`Hello, ${this.name}!`);
} greet.apply(obj); // "Hello, Alice!"
greet.call(obj); // "Hello, Alice!"
const newGreet = greet.bind(obj);
newGreet(); // "Hello, Alice!"

在上面的代码中,我们使用 apply()、call() 和 bind() 方法来将函数的 this 值设置为 obj 对象,并在函数内部使用它。

2. 函数的参数不确定

有时候我们无法确定函数的参数个数,或者我们想要将一个数组作为参数传递给函数。在这种情况下,我们可以使用 apply() 方法来将数组作为参数传递给函数。例如:

function sum(a, b, c) {
return a + b + c;
} const nums = [1, 2, 3]; const result = sum.apply(null, nums); console.log(result); // 6

在上面的代码中,我们使用 apply() 方法将 nums 数组作为参数传递给 sum() 函数,从而计算出它们的和。

3. 继承

在 JavaScript 中,我们可以使用 call() 方法来实现继承。前面讲解对象时也用到过。例如:

function Animal(name) {
this.name = name;
} function Dog(name, breed) {
Animal.call(this, name);
this.breed = breed;
} const myDog = new Dog('Fido', 'Golden Retriever'); console.log(myDog.name); // "Fido"
console.log(myDog.breed); // "Golden Retriever"

在上面的代码中,我们定义了 Animal 和 Dog 两个构造函数,并使用 call() 方法将 Animal 的属性和方法继承到 Dog 中。

4. 部分应用函数

使用 bind() 方法可以实现部分应用函数,即将一个函数的部分参数固定下来,返回一个新的函数。例如:

function multiply(a, b) {
return a * b;
} const double = multiply.bind(null, 2); console.log(double(3)); // 6
console.log(double(4)); // 8

在上面的代码中,我们使用 bind() 方法将 multiply() 函数的第一个参数固定为 2,从而创建了一个新的函数 double(),它的作用是将传入的参数乘以 2。

最后

总之,apply()、call() 和 bind() 方法非常有用,并且在 JavaScript 开发中经常被使用。熟练掌握它们的用法可以帮助我们更好地编写高质量的代码。

理解JS函数之call,apply,bind的更多相关文章

  1. 深入理解JS函数中this指针的指向

    函数在执行时,会在函数体内部自动生成一个this指针.谁直接调用产生这个this指针的函数,this就指向谁. 怎么理解指向呢,我认为指向就是等于.例如直接在js中输入下面的等式: console.l ...

  2. 两个实例轻松理解js函数预解析

    js函数预解析 例子1: 先上一段代码,看看能不能写出最终的执行结果. console.log(a); var a = 1; console.log(a); function a(){ console ...

  3. js函数中的apply()、call()、bind()方法

    ECMAScript中的函数是对象,因此函数也有属性和方法.每个函数都包含两个属性:length和prototype,且每个函数包含两个非继承而来的方法apply()和call().这两个方法的用途都 ...

  4. js中的call,apply,bind区别

    在JavaScript中,call.apply和bind是Function对象自带的三个方法,这三个方法的主要作用是改变函数中的this指向. call.apply.bind方法的共同点和区别:app ...

  5. js对象系列【二】深入理解js函数,详解作用域与作用域链。

    这次说一下对象具体的一个实例:函数,以及其对应的作用域与作用域链.简单的东西大家查下API就行了,这里我更多的是分享自己的理解与技巧.对于作用域和作用域链,相信绝大多数朋友看了我的分享都能基本理解,少 ...

  6. 深入理解JS函数节流和去抖动

    一.什么是节流和去抖? 1.节流 节流就是拧紧水龙头让水少流一点,但是不是不让水流了.想象一下在现实生活中有时候我们需要接一桶水,接水的同时不想一直站在那等着,可能要离开一会去干一点别的事请,让水差不 ...

  7. JS函数与call()apply()详解

    JavaScript中的每个函数都是一个对象. 因为函数都是对象,它们有自己的属性和方法.我们可以把它们看作数据(data). 函数和方法的区别? 函数立足于它们自己(例如:alert()), 而方法 ...

  8. js 中 new call apply bind JSON.stringify 的原理以及模拟实现

    1.new的原理和实现 它创建了一个全新的对象. 它会被执行 [[Prototype]](也就是 __proto__)链接. 它使 this指向新创建的对象. 通过 new创建的每个对象将最终被 [[ ...

  9. 辅助调用函数【call,apply,bind】

    函数也是对象,每个函数都有自己的方法. e.g. var jane = { name:'Jane', sayHelloTo:function(name) { 'use strict'; console ...

  10. 深入理解JS函数作用域链与闭包问题

    function fun(n,o) { console.log(o) return { fun:function(m){ return fun(m,n); } }; } ); a.fun(); a.f ...

随机推荐

  1. 30位以内随机产生时间戳加随机数id

    package com.zx.ps.web.gzdb; import java.text.SimpleDateFormat; import java.util.Date; public class c ...

  2. WeakHashMap 和 HashMap 的区别是什么,何时使用?

    本文已收录到 AndroidFamily,技术和职场问题,请关注公众号 [彭旭锐] 提问. 前言 大家好,我是小彭. 在之前的文章里,我们聊到了 Java 标准库中 HashMap 与 LinkedH ...

  3. Selenium4+Python3系列(十三) - 与docker中的jenkins持续集成

    前言 文章更新到这一篇时,其实我还是很开心的,因为这也正是这系列教程的最后一篇文章,也算是完成了一个阶段性的小目标,也很感谢那些愿意看我文章与我交流学习的同学,感谢有你们的支持和陪伴. Jenkins ...

  4. 使用 System.Text.Json 时,如何处理 Dictionary 中 Key 为自定义类型的问题

    在使用 System.Text.Json 进行 JSON 序列化和反序列化操作时,我们会遇到一个问题:如何处理字典中的 Key 为自定义类型的问题. 背景说明 例如,我们有如下代码:   // 定义一 ...

  5. Jmeter 之 switch 控制器

    switch 控制器作用: switch 控制器起到了分流作用,具体应用在并发时,一部分用户执行某个场景,一部分用户执行另外一种场景,就像吞吐量控制器进行分流操作 switch 控制器字段介绍: 实例 ...

  6. Python AI小项目打包通关:Pyinstaller和Wix都用上了

    最近有个Python小项目要打个包,项目结构比较简单 main.py(主文件), 以及model_050.hdf5 (在云端训练好的AI模型) 主函数里引用了一些包,如下 需要解决的问题: 将main ...

  7. BOM、DOM、两者查找标签的方式和操作标签的方式、操作class和css的方法、事件、jQuery类库

    今日内容 BOM操作 BOM(Browser Object Model)是指浏览器对象模型,它使 JavaScript 有能力与浏览器进行"对话". windows对象 windo ...

  8. Vue+elementui前后端分离,单个图片文件上传和上传时出现的跨域问题的解决方案

    在后端解决跨域问题: 我是通过配置文件来解决跨域问题的 @Configurationpublic class CorsConfig {//解决前后端分离的跨域问题! /** * cors suppor ...

  9. Docker 基础 - 2

    容器操作系统类型 Busybox 集成了一百多个最常用 Linux 命令和工具的软件工具箱. 包含cat echo grep find mount telnet 等 Busybox 是Linux 系统 ...

  10. file过滤器的原理和使用-FileNameFilter过滤器的使用和lambda优化程序

    file过滤器的原理和使用 java.io.FileFilter是一个接口,是File的过滤器.该接口的对象可以传递给File类的listFiles(FileFilter)作为参数,接口中只有一个方法 ...