理解JS函数之call,apply,bind
前言
在 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的更多相关文章
- 深入理解JS函数中this指针的指向
函数在执行时,会在函数体内部自动生成一个this指针.谁直接调用产生这个this指针的函数,this就指向谁. 怎么理解指向呢,我认为指向就是等于.例如直接在js中输入下面的等式: console.l ...
- 两个实例轻松理解js函数预解析
js函数预解析 例子1: 先上一段代码,看看能不能写出最终的执行结果. console.log(a); var a = 1; console.log(a); function a(){ console ...
- js函数中的apply()、call()、bind()方法
ECMAScript中的函数是对象,因此函数也有属性和方法.每个函数都包含两个属性:length和prototype,且每个函数包含两个非继承而来的方法apply()和call().这两个方法的用途都 ...
- js中的call,apply,bind区别
在JavaScript中,call.apply和bind是Function对象自带的三个方法,这三个方法的主要作用是改变函数中的this指向. call.apply.bind方法的共同点和区别:app ...
- js对象系列【二】深入理解js函数,详解作用域与作用域链。
这次说一下对象具体的一个实例:函数,以及其对应的作用域与作用域链.简单的东西大家查下API就行了,这里我更多的是分享自己的理解与技巧.对于作用域和作用域链,相信绝大多数朋友看了我的分享都能基本理解,少 ...
- 深入理解JS函数节流和去抖动
一.什么是节流和去抖? 1.节流 节流就是拧紧水龙头让水少流一点,但是不是不让水流了.想象一下在现实生活中有时候我们需要接一桶水,接水的同时不想一直站在那等着,可能要离开一会去干一点别的事请,让水差不 ...
- JS函数与call()apply()详解
JavaScript中的每个函数都是一个对象. 因为函数都是对象,它们有自己的属性和方法.我们可以把它们看作数据(data). 函数和方法的区别? 函数立足于它们自己(例如:alert()), 而方法 ...
- js 中 new call apply bind JSON.stringify 的原理以及模拟实现
1.new的原理和实现 它创建了一个全新的对象. 它会被执行 [[Prototype]](也就是 __proto__)链接. 它使 this指向新创建的对象. 通过 new创建的每个对象将最终被 [[ ...
- 辅助调用函数【call,apply,bind】
函数也是对象,每个函数都有自己的方法. e.g. var jane = { name:'Jane', sayHelloTo:function(name) { 'use strict'; console ...
- 深入理解JS函数作用域链与闭包问题
function fun(n,o) { console.log(o) return { fun:function(m){ return fun(m,n); } }; } ); a.fun(); a.f ...
随机推荐
- kestrel网络编程--开发Fiddler
1 文章目的 本文讲解基于kestrel开发类似Fiddler应用的过程,让读者了解kestrel网络编程里面的kestrel中间件和http应用中间件.由于最终目的不是输出完整功能的产品,所以这里只 ...
- Python全栈工程师之从网页搭建入门到Flask全栈项目实战(7) - 在线问答系统
1.项目源码/业务逻辑 百度网盘链接:链接:https://pan.baidu.com/s/13VNfrSJE6vcL3HP1J5T8ew 提取码:00s0,项目业务逻辑自行阅读 2.项目搭建 点击新 ...
- ConditionAlOnProperties实现可插拔?
大家好,我是3y,一年CRUD经验用十年的markdown程序员常年被誉为职业八股文选手 我又又又又被吐槽了,随之而来,我的消息推送平台开源项目Austin又又又又更新啦,迭代自己的项目多是一件美事 ...
- [0x11] 130.火车进站问题【卡特兰数】
题意 link(more:129.,P1044) 简化题意:给定严格从 \(1\thicksim n\) 这 \(n(n\leqslant 6\times10^4)\) 个整数,规定每个数都要进出栈各 ...
- 【转载】EXCEL VBA-区域选择
1- 区域命名 ThisWorkbook.Names.Item("foo").RefersTo =Tabelle1!$A$1:$B$1 ThisWorkbook.Names.Ite ...
- AVM 拖动组件 movable-view 介绍
应用开发中拖动功能是比较常见的 ,如滑动解锁,滑动验证码,实现一些小游戏,少儿编程中常见. avm.js 是多端开发框架,一套代码可同时编译为APP .小程序.h5. avm 框架中实现拖动功能非常简 ...
- 工业数据分析为什么要用FusionInsight MRS IoTDB?
摘要:MRS IoTDB,它是华为FusionInsight MRS大数据套件中的时序数据库产品,在深度参与Apache IoTDB社区开源版的基础上推出的高性能企业级时序数据库产品. 本文分享自华为 ...
- UOJ33 [UR#2] 树上 GCD
UOJ33 [UR#2] 树上 GCD 简要题意: 给定一棵有根树,对于每个 \(i \in [1,n)\),求出下式的值: \[Ans[i] = \sum_{u<v} \gcd({\rm{di ...
- python进阶之路19 地狱之门购物车!!!!
地狱之门 # # 项目功能 # 1.用户注册 # 2.用户登录 # 3.添加购物车 # 4.结算购物车 # # 项目说明 # 用户数据采用json格式存储到文件目录db下 一个用户一个单独的文件 # ...
- HHKB Programming Contest 2022 Winter(AtCoder Beginner Contest 282)
前言 好久没有打 AtCoder 了.有点手生.只拿到了 \(\operatorname{rk}1510\),应该上不了多少分. 只切了 \(\texttt{A,B,C,D}\) 四题. A - Ge ...