深入理解JavaScript中的call、apply、bind方法的区别


Posted in Javascript onMay 30, 2016

在JavaScript 中,this的指向是动态变化的,很可能在写程序的过程中,无意中破坏掉this的指向,所以我们需要一种可以把this的含义固定的技术,于是就有了call,apply 和bind这三个方法,来改变函数体内部 this 的指向,因为函数存在「定义时上下文」和「运行时上下文」以及「上下文是可以改变的」这样的概念

apply、call

apply:应用某一对象的一个方法,用另一个对象替换当前对象

call:调用一个对象的一个方法,以另一个对象替换当前对象

function person() {}
 
person.prototype = {
 attr: {age:18,sex:'girl'},
 say: function() {
  console.log("My age is " + this.attr.age);
  console.log("I am a " + this.attr.sex);
 }
}
 
var Marry = new person();
Marry.say();
 
// My age is 18
// I am a girl

改变指向

function person() {}
 
person.prototype = {
 attr: {age:18,sex:'girl'},
 say: function() {
  console.log("My age is " + this.attr.age);
  console.log("I am a " + this.attr.sex);
 }
}
 
xiaoming ={ attr : {age:20,sex:'boy'} };
 
var Marry = new person();
Marry.say();
 
Marry.say.call(xiaoming);
// My age is 18
// I am a girl
// My age is 20
// I am a boy

共同之处

都可以用来代替另一个对象调用一个方法,将一个函数的对象上下文从初始的上下文改变为由 thisObj 指定的新对象。

不同之处

1、apply:最多只能有两个参数——新this对象和一个数组 argArray。如果给该方法传递多个参数,则把参数都写进这个数组里面,当然,即使只有一个参数,也要写进数组里面。如果 argArray 不是一个有效的数组或者不是 arguments 对象,那么将导致一个 TypeError。如果没有提供 argArray 和 thisObj 任何一个参数,那么 Global 对象将被用作 thisObj,并且无法被传递任何参数。

fun.call(thisArg[, arg1[, arg2[, ...]]])
 
function f(x,y){
 console.log(x+y);
}
f.call(null, 1, 1)
//return 2

2、call:则是直接的参数列表,主要用在js对象各方法互相调用的时候,使当前this实例指针保持一致,或在特殊情况下需要改变this指针。如果没有提供 thisObj 参数,那么 Global 对象被用作 thisObj。

fun.apply(thisArg, [argsArray])
 
function f(x,y){
 console.log(x+y);
}
f.call(null, [1,1])
//return 2

apply和call功能一样,只是传入的参数列表形式不同,其中 thisArg 是你想指定的上下文,他可以是任何一个 JavaScript 对象(JavaScript 中一切皆对象),call 需要把参数按顺序传递进去,而 apply 则是把参数放在数组里。

如果某个函数的参数数量是不固定的,当你的参数是明确知道数量时用 call ,而不确定的时候用 apply,然后把参数 push 进数组传递进去。当参数数量不确定时,函数内部也可以通过 arguments 这个数组来遍历所有的参数,我们看一下一些用法

代码一

var array1 = [12 , "foo" , {name:"Joe"} , -2458];
var array2 = ["Doe" , 555 , 100];
Array.prototype.push.apply(array1, array2);
console.log(array1);
//[12, "foo", Object, -2458, "Doe", 555, 100]

代码二

var numbers = [5, 458 , 120 , -215 ];
Math.max.apply(null,numbers);
//458

代码三

log(12 , "foo" , {name:"Joe"} , -2458);
function log(){
 var args = Array.prototype.slice.call(arguments);
 args.unshift('(app)');
 console.log.apply(console, args);
};
// (app) 12 foo Object {name: "Joe"} -2458

bind

fun.bind(thisArg[, arg1[, arg2[, ...]]])

与上面不同的是,bind会返回一个改变this指向的新函数 ,注意这里强调的是新函数,其与之前的使用的不是同一块内存地址,所以当你需要重复使用这个函数的时候,你就不得不把其保存到一个变量,方便下次调用。上面的两个函数都是返回的执行结果,即调用即执

行,此外,另外需要注意的地方是,bind函数中的首个参数,会自动成为返回新函数中参数的默认值,那么正式调用的时候,只需要给出除首个参数外,剩余参数即可。

function f(x,y){
 console.log(x+y);
}
f.call(null, [1,1])
var new_f = f.bind(null,1,1)
//return new function
new_f(1)
//return 2

需要说明的是,上面所有示例代码中的thisArg参数均用null来代替了,在未给出指定thisArg对象的情况下,null与undefined下this指向是全局对象,即js代码执行环境

apply、call、bind比较

var obj = {bar: 'Oops , this is a bad idea' };
 
var foo = {
  get: function() {
    return this.bar;
  }
}
 
var bind = foo.get.bind(obj) ,call = foo.get.call(obj) ,apply = foo.get.apply(obj);
 
console.log(bind(),call,apply);
console.log(bind,call,apply); 
 
console.log(typeof bind,typeof call,typeof apply);
console.log(typeof bind(),typeof call,typeof apply);

深入理解JavaScript中的call、apply、bind方法的区别

看到区别没有,区别是,当你希望改变上下文环境之后并非立即执行,而是回调执行的时候,使用 bind() 方法。而 apply/call 则会立即执行函数

apply 、 call 、bind 三者都是用来改变函数的this对象的指向的;
apply 、 call 、bind 三者第一个参数都是this要指向的对象,也就是想指定的上下文;
apply 、 call 、bind 三者都可以利用后续参数传参;
bind是返回对应函数,便于稍后调用;apply、call则是立即调用

以上这篇深入理解JavaScript中的call、apply、bind方法的区别就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
javascript web对话框与弹出窗口
Feb 22 Javascript
Ext javascript建立超链接,进行事件处理的实现方法
Mar 22 Javascript
zTree插件之单选下拉菜单实例代码
Nov 07 Javascript
js字符串转成JSON
Nov 07 Javascript
javascript类型转换使用方法
Feb 08 Javascript
jQuery中使用data()方法读取HTML5自定义属性data-*实例
Apr 11 Javascript
jQuery插件kinMaxShow扩展效果用法实例
May 04 Javascript
angularJS Provider、factory、service详解及实例代码
Sep 21 Javascript
js继承实现方法详解
Dec 16 Javascript
vue-cli 为项目设置别名的方法
Oct 15 Javascript
viewer.js实现图片预览功能
Jun 24 Javascript
实例讲解React 组件生命周期
Jul 08 Javascript
全面解析Bootstrap中transition、affix的使用方法
May 30 #Javascript
全面解析Bootstrap中form、navbar的使用方法
May 30 #Javascript
js实现页面a向页面b传参的方法
May 29 #Javascript
浅析jQuery中使用$所引发的问题
May 29 #Javascript
基于jQuery实现仿百度首页选项卡切换效果
May 29 #Javascript
jQuery实现图像旋转动画效果
May 29 #Javascript
jQuery中使用animate自定义动画的方法
May 29 #Javascript
You might like
用PHP和ACCESS写聊天室(三)
2006/10/09 PHP
php获取表单中多个同名input元素的值
2014/03/20 PHP
PHP中substr()与explode()函数用法分析
2014/11/24 PHP
phpstorm最新激活码分享亲测phpstorm2020.2.3版可用
2020/11/22 PHP
文本框水印提示效果的简单实现代码
2014/02/22 Javascript
js+HTML5基于过滤器从摄像头中捕获视频的方法
2015/06/16 Javascript
jQuery mobile的header和footer在点击屏幕的时候消失的解决办法
2016/07/01 Javascript
手机浏览器 后退按钮强制刷新页面方法总结
2016/10/09 Javascript
vue.js通过自定义指令实现数据拉取更新的实现方法
2016/10/18 Javascript
vue-resourse将json数据输出实例
2017/03/08 Javascript
利用n工具轻松管理Node.js的版本
2017/04/21 Javascript
详解Vue.use自定义自己的全局组件
2017/06/14 Javascript
Angular 2父子组件之间共享服务通信的实现
2017/07/04 Javascript
Javascript实现鼠标点击冒泡特效
2019/12/24 Javascript
微信小程序12行js代码自己写个滑块功能(推荐)
2020/07/15 Javascript
vant picker+popup 自定义三级联动案例
2020/11/04 Javascript
[01:25:09]2014 DOTA2国际邀请赛中国区预选赛 5 23 CIS VS DT第二场
2014/05/24 DOTA
[02:00]DAC2018主宣传片——龙征四海,剑问东方
2018/03/20 DOTA
python中尾递归用法实例详解
2015/04/28 Python
Python利用turtle库绘制彩虹代码示例
2017/12/20 Python
梅尔倒谱系数(MFCC)实现
2019/06/19 Python
Python单元测试工具doctest和unittest使用解析
2019/09/02 Python
Python3爬虫中Ajax的用法
2020/07/10 Python
使用Django的JsonResponse返回数据的实现
2021/01/15 Python
python中添加模块导入路径的方法
2021/02/03 Python
HTML5 video循环播放多个视频的方法步骤
2020/08/06 HTML / CSS
美国汽车性能部件和赛车零件网站:Vivid Racing
2018/03/27 全球购物
智能家居、吸尘器、滑板车、电动自行车网上购物:Geekmaxi
2021/01/18 全球购物
毕业生自荐书模版
2014/01/04 职场文书
求职信的七个关键技巧
2014/02/05 职场文书
出纳会计岗位职责
2014/03/12 职场文书
高等教育专业自荐信范文
2014/03/26 职场文书
三方协议书范本
2014/04/22 职场文书
就业意向书范本
2015/05/11 职场文书
退休教师追悼词
2015/06/23 职场文书
logback 实现给变量指定默认值
2021/08/30 Java/Android