深入理解关于javascript中apply()和call()方法的区别


Posted in Javascript onApril 12, 2016

如果没接触过动态语言,以编译型语言的思维方式去理解javaScript将会有种神奇而怪异的感觉,因为意识上往往不可能的事偏偏就发生了,甚至觉得不可理喻.如果在学JavaScript这自由而变幻无穷的语言过程中遇到这种感觉,那么就从现在形始,请放下的您的”偏见”,因为这对您来说绝对是一片新大陆,让JavaScrip慢慢融化以前一套凝固的编程意识,注入新的生机!

好,言归正传,先理解JavaScrtipt动态变换运行时上下文特性,这种特性主要就体现在apply, call两个方法的运用上.

区分apply,call就一句话,

foo.call(this, arg1,arg2,arg3) == foo.apply(this, arguments)==this.foo(arg1, arg2, arg3)

call, apply都属于Function.prototype的一个方法,它是JavaScript引擎内在实现的,因为属于Function.prototype,所以每个Function对象实例,也就是每个方法都有call, apply属性.既然作为方法的属性,那它们的使用就当然是针对方法的了.这两个方法是容易混淆的,因为它们的作用一样,只是使用方式不同.

相同点: 两个方法产生的作用是完全一样的

不同点: 方法传递的参数不同

那什么是方法产生的作用,方法传递的参数是什么呢?

我们就上面的foo.call(this, arg1, arg2, arg3)展开分析.

foo是一个方法,this是方法执行时上下文相关对象,arg1, arg2, arg3是传给foo方法的参数.这里所谓的方法执行时上下文相关对象, 如果有面向对象的编程基础,那很好理解,就是在类实例化后对象中的this.

在JavaScript中,代码总是有一个上下文对象,代码处理该对象之内. 上下文对象是通过this变量来体现的, 这个this变量永远指向当前代码所处的对象中.

为了更好的领会这this是什么,举个例子.

/创建一个A类
function A(){
//类实例化时将运行以下代码
//此时的执行上下文对象就是this,就是当前实例对象
this.message = “message of a”;
this.getMessage = function(){
<SPAN style="WHITE-SPACE: pre">	</SPAN>return this.message;
<SPAN style="WHITE-SPACE: pre">	</SPAN>}
}
//创建一个A类实例对象
var a = new A();
//调用类实例getMessage方法获得message值
alert(a.getMessage());
//创建一个B类
function B(){
this.message = ”message of b”;
this.setMessage = function(msg){
<SPAN style="WHITE-SPACE: pre">	</SPAN>this.message = msg;
<SPAN style="WHITE-SPACE: pre">	</SPAN>}
}
//创建一个B类实例对象
var a = new B();

可见,A, B类都有一个message属性(面向对象中所说的成员),A有获取消息的getMessage方法,B有设置消息的setMessage方法,下面来显示call的威力.

//给对象a动态指派b的setMessage方法,注意,a本身是没有这方法的!
b.setMessage.call(a, “a的消息”);
//下面将显示”a的消息”
alert(a.getMessage());
//给对象b动态指派a的getMessage方法,注意,b本身也是没有这方法的!

这就是动态语言 JavaScript call的威力所在!

简直是”无中生有”,对象的方法可以任意指派,而对象本身一直都是没有这方法的,注意是指派,通俗点就是,方法是借给另一个对象的调用去完成任务,原理上是方法执行时上下文对象改变了.

所以 b.setMessage.call(a, “a的消息”); 就等于用a作执行时上下文对象调用b对象的setMessage方法,而这过程中与b一点关系都没有, 作用等效于a.setMessage( “a的消息”);

因为apply与call产生的作用是一样的,可以说

call, apply作用就是借用别人的方法来调用,就像调用自己的一样.

好,理解了call, apply相同处—?作用后,再来看看它们的区别,看过上面例子,相信您大概知道了.

从 b.setMessage.call(a, “a的消息”) 等效于 a.setMessage( “a的消息”) 可以看出, “a的消息”在call中作为一个参数传递,

那么在apply中是怎么表示的呢,直接解释说不清楚,apply要结合应用场景才一目了然.我们来设计一个应用场景:

function print(a, b, c, d){
alert(a + b + c + d);
}
function example(a, b , c , d){
//用call方式借用print,参数显式打散传递
print.call(this, a, b, c, d);
//用apply方式借用print, 参数作为一个数组传递,
//这里直接用JavaScript方法内本身有的arguments数组
print.apply(this, arguments);
//或者封装成数组
print.apply(this, [a, b, c, d]);
}
//下面将显示”背光脚本”
example(”背” , “光” , “脚”, “本”);

在这场景中, example方法内,a, b, c, d作为方法传递的参数, 方法分别运用了apply, call去借print方法来调用,

最后一句由于直接调用example方法, 所以在该方法中的上下文对象this就是window对象.

所以,call, apply方法它们除了第一个参数,即执行时上下文对象相同外,call方法的其它参数将依次传递给借用的方法作参数,而apply就两个参数,第二个参数为一个数组传递.所以可以说成

call, apply方法区别是,从第二个参数起, call方法参数将依次传递给借用的方法作参数, 而apply直接将这些参数放到一个数组中再传递, 最后借用方法的参数列表是一样的.

 
应用场景:

当参数明确时可用call, 当参数不明确时可用apply给合arguments

//例
print.call(window, “背” , “光” , “脚”, “本”);
//foo参数可能为多个
function foo(){
<SPAN style="WHITE-SPACE: pre">	</SPAN>print.apply(window, arguments);
}

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

Javascript 相关文章推荐
js 页面刷新location.reload和location.replace的区别小结
Dec 24 Javascript
jQuery选择没有colspan属性的td的代码
Jul 06 Javascript
[原创]JQuery 在表单提交之前修改 提交的值
Apr 14 Javascript
使用jQuery的load方法设计动态加载及解决被加载页面js失效问题
Mar 01 Javascript
vue使用stompjs实现mqtt消息推送通知
Jun 22 Javascript
微信小程序使用picker实现时间和日期选择框功能【附源码下载】
Dec 11 Javascript
深入浅析Vue全局组件与局部组件的区别
Jun 15 Javascript
Angular7.2.7路由使用初体验
Mar 01 Javascript
Vue 利用指令实现禁止反复发送请求的两种方法
Sep 15 Javascript
使用webpack搭建pixi.js开发环境
Feb 12 Javascript
js实现登录时记住密码的方法分析
Apr 05 Javascript
解决vue中axios设置超时(超过5分钟)没反应的问题
Sep 04 Javascript
javascript中去除数组重复元素的实现方法【实例】
Apr 12 #Javascript
Bootstrap每天必学之折叠
Apr 12 #Javascript
JavaScript几种数组去掉重复值的方法推荐
Apr 12 #Javascript
学习使用bootstrap基本控件(table、form、button)
Apr 12 #Javascript
如何高效率去掉js数组中的重复项
Apr 12 #Javascript
学习使用bootstrap3栅格系统
Apr 12 #Javascript
分享两段简单的JS代码防止SQL注入
Apr 12 #Javascript
You might like
PHP的FTP学习(一)
2006/10/09 PHP
PHP关联链接常用代码
2012/11/05 PHP
php与flash as3 socket通信传送文件实现代码
2014/08/16 PHP
ecshop 2.72如何修改后台访问地址
2015/03/03 PHP
利用PHP_XLSXWriter代替PHPExcel的方法示例
2017/07/16 PHP
javascript对象的property和prototype是这样一种关系
2007/03/24 Javascript
日期 时间js控件
2009/05/07 Javascript
对jQuery的事件绑定的一些思考(补充)
2013/04/20 Javascript
js字符串截取函数substr substring slice使用对比
2013/11/27 Javascript
JavaScript判断变量是否为空的自定义函数分享
2015/01/31 Javascript
使用jQuery mobile库检测url绝对地址和相对地址的方法
2015/12/04 Javascript
微信小程序 animation API详解及实例代码
2016/10/08 Javascript
将鼠标焦点定位到文本框最后(代码分享)
2017/01/11 Javascript
JS正则获取HTML元素的方法
2017/03/31 Javascript
jquery ajaxfileuplod 上传文件 essyui laoding 效果【防止重复上传文件】
2018/05/26 jQuery
NodeJS如何实现同步的方法示例
2018/08/24 NodeJs
react native基于FlatList下拉刷新上拉加载实现代码示例
2018/09/30 Javascript
js 实现 list转换成tree的方法示例(数组到树)
2019/08/18 Javascript
浅谈vant组件Picker 选择器选单选问题
2020/11/04 Javascript
可用于监控 mysql Master Slave 状态的python代码
2013/02/10 Python
python实现进程间通信简单实例
2014/07/23 Python
详细介绍Ruby中的正则表达式
2015/04/10 Python
python实现多线程的两种方式
2016/05/22 Python
python输入整条数据分割存入数组的方法
2018/11/13 Python
Python学习笔记之Django创建第一个数据库模型的方法
2019/08/07 Python
后端开发使用pycharm的技巧(推荐)
2020/03/27 Python
Pycharm安装python库的方法
2020/11/24 Python
css3实现3D色子翻转特效
2014/12/23 HTML / CSS
PHP开发的一般流程
2013/08/13 面试题
软件测试英文面试题
2012/10/14 面试题
python+selenium小米商城红米K40手机自动抢购的示例代码
2021/03/24 Python
实习老师个人总结的自我评价
2013/09/28 职场文书
大学本科毕业生的自我鉴定范文
2013/11/19 职场文书
离婚协议书应该怎么写
2014/10/12 职场文书
运动会闭幕式通讯稿
2015/07/18 职场文书
MySQL创建管理LIST分区
2022/04/13 MySQL