前言

在 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. C++ 之 宏定义

    宏在 C 语言中非常重要,但在 C++ 中却无甚大用,普遍的共识:尽量避免使用宏 C++ 之父 Bjarne 在<C++ Programming Language>中写到 Avoid ma ...

  2. Springboot 2.3.1配置拦截器遇到的坑

    1.多个配置类继承WebConfigureSupport或实现WebConfigure接口导致拦截器失效 2.拦截器中的bean无法正常注入,SpringBoot项目的Bean装配默认规则是根据App ...

  3. MySQL进阶实战3,mysql索引详解,上篇

    一.索引 索引是存储引擎用于快速查找记录的一种数据结构.我觉得数据库中最重要的知识点,就是索引. 存储引擎以不同的方式使用B-Tree索引,性能也各有不同,各有优劣.例如MyISAM使用前缀压缩技术使 ...

  4. PHP8.1.10手动安装教程及报错解决梳理

    安装教程参考一:https://www.cnblogs.com/haw2106/p/9839655.html 安装教程参考二:https://www.cnblogs.com/jiangfeilong/ ...

  5. [OpenCV实战]42 数码单反相机的技术细节

    在这篇文章中,我们将说明数码单反相机DSLR(Digital Single Lens Reflex Camera)的各个技术方面.本文将说明焦距(focal length),f-stop,景深(dep ...

  6. 基于docker容器的MySQL主从设置及efcore读写分离

    1.基于docker部署MySQL,设置主从 本操作基于已经拉取的镜像(docker pull mysql) 创建一主一从两个数据库容器 docker run -d -p 3307:3306 -e M ...

  7. python数据分析与可视化【思维导图】

    python数据分析与可视化常用库 numpy+matplotlib+pandas 思维导图 图中难免有错误,后期随着学习与应用的深入,会不断修改更新. 当前版本号:1.0 numpy介绍 NumPy ...

  8. python之路56 dajngo最后一天 csrf跨站请求 auth模块登录注册方法

    csrf跨站请求伪造 钓鱼网站:模仿一个正规的网站 让用户在该网站上做操作 但是操作的结果会影响到用户正常的网站账户 但是其中有一些猫腻 eg:英语四六级考试需要网上先缴费 但是你会发现卡里的钱扣了但 ...

  9. 8. 字符串转整数 (atoi)

    题目 代码 class Solution { public: int myAtoi(string str) { int res=0,sign=1; int i=str.find_first_not_o ...

  10. for循环 rang方法

    今日内容 while循环补充说明 1.死循环 真正的死循环是一旦执行 cpu的功耗急剧上升 直到系统采取紧急措施 2.嵌套及全局标志位 强调: 一个break只能结束他所在那一层的循环 如果想一次性结 ...