一个简单的例子让你很轻松地明白JavaScript中apply、call、bind三者的用法及区别
JavaScript中apply、call、bind三者的用法及区别
引言
这是我的第一篇前端面试题文章,也是用来帮助自己巩固复习知识, 在之后希望拿到自己满意的offer,这第一篇讲的就是一道常见的面试题,关于JavaScript中apply、call、bind三者的用法及区别。
正文
如果有不想看过程的小伙伴可以直接跳到 总结
一、apply、call、bind的共同用法
先将一下特别概念性的,三者的共同用法就是可以改变函数的this指向,将函数绑定到上下文中。接下来看一个普通的应用场景:
let obj1 = {
my_favorite : 'banana',
add: function (fruit) {
console.log('我喜欢的水果是' + fruit + ',但是我最爱的水果是' + this.my_favorite)
}
}
let obj2 = {
my_favorite: 'orange'
}
obj1.add('apple')
//输出结果为: 我喜欢的水果是apple,但是我最爱的水果是banana
可以看到最后一行代码我们调用了obj1中的add函数,并传入了一个参数apple。add函数中的 this 指的是他所在的对象obj1,所以this.my_favorite 就是banana, 但是我们如果想获得obj2中的my_favorite, 该怎么办呢?接下来看一下每个函数的用法吧~
二、 apply
语法:fun.apply(obj,[arry])
- 第一个参数: 传入的是this需要指向的对象
- 第二个参数: 传入的是一个数组,作为 fun 函数的参数
let obj1 = {
my_favorite : 'banana',
add: function (...fruit) { //...fruit 表示的意思是可接收多个参数
console.log('我喜欢的水果是' + fruit + ',但是我最爱的水果是' + obj2.my_favorite)
}
}
let obj2 = {
my_favorite: 'orange'
}
obj1.add.apply(obj2, ['apple', 'Watermelon'])
//输出结果为:我喜欢的水果是apple,Watermelon,但是我最爱的水果是orange
obj1.add.apply(obj2, ['apple', 'Watermelon'])
这一行代码, 第一个参数将obj1中的add函数的 this 指向了obj2 , 第二个参数以数组形式传入多个参数,作为 obj1 中的 add 函数传入的参数, 所以最后能将 apple 和 watermelon 都输出。
三、 call
语法:fun.call(obj,[arg1, arg2, arg3...,argN]
- 第一个参数: 传入的是this需要指向的对象
- 其余参数: 除了第一个参数,其他的参数都是作为 fun 的传入参数
call 和 apply 唯一的区别就是在于给 fun 函数传入参数的形式不同, call是将多个参数逐个传入, apply 是 将多个参数放在一个数组中,一起传入
话不多说,直接来看看 call 的例子:
let obj1 = {
my_favorite : 'banana',
add: function (...fruit) {
console.log('我喜欢的水果是' + fruit + ',但是我最爱的水果是' + obj2.my_favorite)
}
}
let obj2 = {
my_favorite: 'orange'
}
obj1.add.call(obj2, 'apple', 'Watermelon')
//输出结果为:我喜欢的水果是apple,Watermelon,但是我最爱的水果是orange
大家可以对比一下 call 和 apply 的区别到底在哪里
四、 bind
语法:fun.bind(obj)(arg)
- bind的第一个参数:传入的是this需要指向的对象
- 因为函数bind返回的是一个函数,即将 fun 的 this 指向 obj 后不进行自身调用, 我们需要自己进行一次调用, 所以要在函数bind后面再加一个小括号进行自身调用, 而小括号里则可以输出传给 fun 函数的参数, 输入的可以是数组形式的,也可以是逐个传入的。
简单看一下例子:
let obj1 = {
my_favorite : 'banana',
add: function (...fruit) {
console.log('我喜欢的水果是' + fruit + ',但是我最爱的水果是' + obj2.my_favorite)
}
}
let obj2 = {
my_favorite: 'orange'
}
obj1.add.bind(obj2)('apple', 'Watermelon')
obj1.add.bind(obj2)(['apple', 'Watermelon'])
//输出结果为:我喜欢的水果是apple,Watermelon,但是我最爱的水果是orange
//输出结果为:我喜欢的水果是apple,Watermelon,但是我最爱的水果是orange
可以很清楚的看到, bind 有点类似 apply 和 call 的结合,只不过它返回的是一个函数,需要自己进行一下调用, 而传给 fun 函数的参数形式可以是像 apply 一样的数组形式, 也可以是像 call 一样的逐个传入的形式。
五、其他应用场景
- 获取一个数组中的最大值或最小值
/* 普通情况,我们是这样获取数组中的最大值、最小值的 */
let max = Math.max(22, 33, 54, 88, 35, 44, 78, 80)
let min = Math.min(22, 33, 54, 88, 35, 44, 78, 80)
console.log(max, min)
//88 22
有没有感觉特别的麻烦,那接下来看看使用了这几个函数之后是什么样子的吧
/* 使用函数apply */
let arr = [22, 33, 54, 88, 35, 44, 78, 80]
let max = Math.max.apply(null, arr)
let min = Math.min.apply(null, arr)
console.log(max, min)
//88 22
这样看着是不是特别的简洁,因为我们不需要把arr 中的数组一个个传入进行比较,而是直接以数组形式传入就可以了。
六、总结
- apply、call、bind 三者都是可以改变函数的this指向的
- apply 和 call 都是改变函数this指向,并传入参数后立即调用执行该函数
- bind 是在改变函数this指向,并传入参数后返回一个新的函数,不会立即调用执行
- apply 传入的参数是数组形式的;call 传入的参数是按顺序的逐个传入并以逗号隔开; bind 传入的参数既可以是数组形式,也可以是按顺序逐个传入。
结束语
apply、call、bind三者的用法及区别是经常考到的一道面试题,希望对大家有所帮助。之后我还会每天更新一个前端面试题,来帮助大家以及我自己巩固知识点,喜欢的小伙伴可以点个关注, 希望你们都拿到自己满意的offer
一个简单的例子让你很轻松地明白JavaScript中apply、call、bind三者的用法及区别的更多相关文章
- 用一个简单的例子来理解python高阶函数
============================ 用一个简单的例子来理解python高阶函数 ============================ 最近在用mailx发送邮件, 写法大致如 ...
- Spring-Context之一:一个简单的例子
很久之前就想系统的学习和掌握Spring框架,但是拖了很久都没有行动.现在趁着在外出差杂事不多,就花时间来由浅入深的研究下Spring框架.Spring框架这几年来已经发展成为一个巨无霸产品.从最初的 ...
- 关于apriori算法的一个简单的例子
apriori算法是关联规则挖掘中很基础也很经典的一个算法,我认为很多教程出现大堆的公式不是很适合一个初学者理解.因此,本文列举一个简单的例子来演示下apriori算法的整个步骤. 下面这个表格是代表 ...
- 跨站脚本功攻击,xss,一个简单的例子让你知道什么是xss攻击
跨站脚本功攻击,xss,一个简单的例子让你知道什么是xss攻击 一.总结 一句话总结:比如用户留言功能,用户留言中写的是网页可执行代码,例如js代码,然后这段代码在可看到这段留言的不同一户的显示上就会 ...
- 从一个简单的例子谈谈package与import机制
转,原文:http://annie09.iteye.com/blog/469997 http://blog.csdn.net/gdsy/article/details/398072 这两篇我也不知道到 ...
- 扩展Python模块系列(二)----一个简单的例子
本节使用一个简单的例子引出Python C/C++ API的详细使用方法.针对的是CPython的解释器. 目标:创建一个Python内建模块test,提供一个功能函数distance, 计算空间中两 ...
- fitnesse - 一个简单的例子(slim)
fitnesse - 一个简单的例子(slim) 2017-09-30 目录1 编写测试代码(Fixture code)2 编写wiki page并运行 2.1 新建wikiPage 2.2 运行 ...
- Struts2的配置和一个简单的例子
Struts2的配置和一个简单的例子 笔记仓库:https://github.com/nnngu/LearningNotes 简介 这篇文章主要讲如何在 IntelliJ IDEA 中使用 Strut ...
- 一个简单的例子搞懂ES6之Promise
ES5中实现异步的常见方式不外乎以下几种: 1. 回调函数 2. 事件驱动 2. 自定义事件(根本上原理同事件驱动相同) 而ES6中的Promise的出现就使得异步变得非常简单.promise中的异步 ...
随机推荐
- 细说websocket快速重连机制
文|马莹莹 网易智慧企业web前端开发工程师 引言 在一个完善的即时通讯应用中,websocket是极其关键的一环,它为web应用的客户端和服务端提供了一种全双工的通信机制,但由于它本身以及其底层依赖 ...
- node学习第一天
创建服务器 利用require引入http模块:var http=require("http") 利用http模块创建server服务器; 创建服务器:var server=htt ...
- 前端学习(一):Html
进击のpython ***** 前端学习--HTML HTML全称HyperText Mackeup Language,超文本标记语言 网页的超链接,图片,音频,视频都可以超文本 标记就相当于你在本子 ...
- 笑谈AlphaGo对战人类从模仿到超越之奥义
起源 故事从一个围棋说起,人类赖以自豪的智慧,为什么这么说,因为据统计,从这19乘19的方格中摆棋,可以有10的340次方的可能,用一个更形象的比拟,全宇宙的原子加起来,也比这个数要小. 也就是说,如 ...
- c++ string 类型 大小写转换
还是用以前的库函数就行的,toupper(int c)小写变大写和tolower(int c)大写变小写 可以直接这么干 string s = "ABCDEFG"; for( in ...
- Python math 模块、cmath 模块
Python math 模块.cmath 模块 Python 中数学运算常用的函数基本都在 math 模块.cmath 模块中.高佣联盟 www.cgewang.com Python math 模块提 ...
- Android JNI之编译
JNI代码都写好了,在编译之前我们有非常重要的一部,就是写mk文件,mk文件就相当于gcc编译时的Makefile文件,它是用来告诉编译器如何去编译的. 这里只对自己理解和常用的知识点做记录,想要看关 ...
- 网络安全 - SSL/TLS协议运行机制的概述
大学时也系统学过相关的网络安全,但那时并没有理论联系实践,稀里糊涂的,现在才意识到所学的东西都是好东西,可惜已晚. 来自http://www.ruanyifeng.com/blog/2014/02/s ...
- PyCharm2020激活破解教程
本文内容皆为作者原创,如需转载,请注明出处:https://www.cnblogs.com/xuexianqi/p/12767075.html 免责声明:本方法只做学习研究之用,不得用于商业用途 若经 ...
- linux的文件处理(匹配 正则表达式 egrep awk sed)和系统、核心数据备份
文件处理 1.处理方式 匹配 正则表达式 egrep awk sed 2.文件中的处理字符 \n 新行符 换行 \t 制表符 tab键 缺省8个空格 \b 退格符 backspace键 退格键 ...