JavaScript中的call和apply的用途以及区别


Posted in Javascript onJanuary 11, 2017

apply 接受两个参数,第一个参数指定了函数体内this 对象的指向,第二个参数为一个带下标的集合,这个集合可以为数组,也可以为类数组,apply 方法把这个集合中的元素作为参数传递给被调用的函数:

var func = function( a, b, c ){
  alert ( [ a, b, c ] ); // 输出 [ 1, 2, 3 ]
};
func.apply( null, [ 1, 2, 3 ] );

在这段代码中,参数 1、2、3 被放在数组中一起传入func函数,它们分别对应func参数列表中的a、b、c。

call 传入的参数数量不固定,跟apply 相同的是,第一个参数也是代表函数体内的this 指向,从第二个参数开始往后,每个参数被依次传入函数:

var func = function( a, b, c ){
  alert ( [ a, b, c ] ); // 输出 [ 1, 2, 3 ]
};
func.call( null, 1, 2, 3 );

当调用一个函数时,JavaScript 的解释器并不会计较形参和实参在数量、类型以及顺序上的区别,JavaScript 的参数在内部就是用一个数组来表示的。从这个意义上说,apply比call的使用率更高,我们不必关心具体有多少参数被传入函数,只要用apply 一股脑地推过去就可以了。call是包装在apply上面的一颗语法糖,如果我们明确地知道函数接受多少个参数,而且想一目了然地表达形参和实参的对应关系,那么也可以用call 来传送参数。

call和apply的用途

1. 改变this 指向

call 和apply 最常见的用途是改变函数内部的this 指向,我们来看个例子:

var obj1 = {
  name: 'sven'
};
var obj2 = {
  name: 'anne'
};
window.name = 'window';
var getName = function(){
  alert ( this.name );
};
getName(); // 输出: window
getName.call( obj1 ); // 输出: sven
getName.call( obj2 ); // 输出: anne

当执行getName.call( obj1 )这句代码时,getName 函数体内的this 就指向obj1 对象,所以此处的

var getName = function(){
alert ( this.name );
};

实际上相当于:

var getName = function(){
alert ( obj1.name ); // 输出: sven
};

在实际开发中,经常会遇到this指向被不经意改变的场景,比如有一个div节点,div节点的onclick 事件中的this 本来是指向这个div的:

document.getElementById( 'div1' ).onclick = function(){
  alert( this.id ); // 输出:div1
};

假如该事件函数中有一个内部函数func,在事件内部调用func 函数时,func 函数体内的this就指向了window,而不是我们预期的div,见如下代码:

document.getElementById( 'div1' ).onclick = function(){
  alert( this.id ); // 输出:div1
  var func = function(){
    alert ( this.id ); // 输出:undefined
  }
  func();
};

这时候我们用call 来修正func 函数内的this,使其依然指向div:

document.getElementById( 'div1' ).onclick = function(){
  var func = function(){
    alert ( this.id ); // 输出:div1
  }
  func.call( this );
};

2. Function.prototype.bind

大部分高级浏览器都实现了内置的Function.prototype.bind,用来指定函数内部的this 指向,即使没有原生的Function.prototype.bind 实现,我们来模拟一个也不是难事,代码如下:

Function.prototype.bind = function( context ){
var self = this; // 保存原函数
return function(){ // 返回一个新的函数
    return self.apply( context, arguments ); // 执行新的函数的时候,会   把之前传入的context
  // 当作新函数体内的this
  }
};
var obj = {
  name: 'sven'
};
var func = function(){
  alert ( this.name ); // 输出:sven
}.bind( obj);
func();

我们通过Function.prototype.bind 来“包装”func 函数,并且传入一个对象context 当作参数,这个context 对象就是我们想修正的this 对象。

在Function.prototype.bind 的内部实现中,我们先把func 函数的引用保存起来,然后返回一个新的函数。当我们在将来执行func 函数时,实际上先执行的是这个刚刚返回的新函数。在新函数内部,self.apply( context, arguments )这句代码才是执行原来的func 函数,并且指定context对象为func 函数体内的this。

这是一个简化版的Function.prototype.bind 实现,通常我们还会把它实现得稍微复杂一点,

使得可以往func 函数中预先填入一些参数:

Function.prototype.bind = function(){
  var self = this, // 保存原函数
  context = [].shift.call( arguments ), // 需要绑定的this 上下文
  args = [].slice.call( arguments ); // 剩余的参数转成数组
  return function(){ // 返回一个新的函数
    return self.apply( context, [].concat.call( args, [].slice.call(  arguments ) ) );
    // 执行新的函数的时候,会把之前传入的context 当作新函数体内的this
    // 并且组合两次分别传入的参数,作为新函数的参数
  }
};
var obj = {
  name: 'sven'
};
var func = function( a, b, c, d ){
  alert ( this.name ); // 输出:sven
  alert ( [ a, b, c, d ] ) // 输出:[ 1, 2, 3, 4 ]
}.bind( obj, 1, 2 );
func( 3, 4 );

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持三水点靠木!

Javascript 相关文章推荐
JavaScript 全角转半角部分
Oct 28 Javascript
jquery 事件对象属性小结
Apr 27 Javascript
动态创建样式表在各浏览器中的差异测试代码
Sep 13 Javascript
SOSO地图API使用(一)在地图上画圆实现思路与代码
Jan 15 Javascript
select标签模拟/美化方法采用JS外挂式插件
Apr 01 Javascript
Bootstrap CSS组件之输入框组
Dec 17 Javascript
jquery append与appendTo方法比较
May 24 jQuery
浅谈angular4 ng-content 中隐藏的内容
Aug 18 Javascript
使用JavaScript实现在页面中显示距离2017年中秋节的天数
Sep 26 Javascript
vue 中directive功能的简单实现
Jan 05 Javascript
解决layui 复选框等内置控件不显示的问题
Aug 14 Javascript
逐行分析鸿蒙系统的 JavaScript 框架(推荐)
Sep 17 Javascript
Vue数据驱动模拟实现2
Jan 11 #Javascript
jQuery实现对象转为url参数的方法
Jan 11 #Javascript
将鼠标焦点定位到文本框最后(代码分享)
Jan 11 #Javascript
移动端界面的适配
Jan 11 #Javascript
bootstrapValidator bootstrap-select验证不可用的解决办法
Jan 11 #Javascript
HTML5 js实现拖拉上传文件功能
Nov 20 #Javascript
Bootstrap表单控件使用方法详解
Jan 11 #Javascript
You might like
一个好用的分页函数
2006/11/16 PHP
wamp下修改mysql访问密码的解决方法
2013/05/07 PHP
php上传图片之时间戳命名(保存路径)
2014/08/15 PHP
解决微信授权回调页面域名只能设置一个的问题
2016/12/11 PHP
thinkPHP5.0框架自动加载机制分析
2017/03/18 PHP
PHP实现数组的笛卡尔积运算示例
2017/12/15 PHP
表单元素事件 (Form Element Events)
2009/07/17 Javascript
给ListBox添加双击事件示例代码
2013/12/02 Javascript
js实现按钮控制图片360度翻转特效的方法
2015/02/17 Javascript
javascript中Array()数组函数详解
2015/08/23 Javascript
详解javascript传统方法实现异步校验
2016/01/22 Javascript
Radio 单选JS动态添加的选项onchange事件无效的解决方法
2016/12/12 Javascript
JavaScript实现跟随滚动缓冲运动广告框
2017/07/15 Javascript
利用adb shell和node.js实现抖音自动抢红包功能(推荐)
2018/02/22 Javascript
webpack@v4升级踩坑(小结)
2018/10/08 Javascript
教你如何编写Vue.js的单元测试的方法
2018/10/17 Javascript
微信小程序使用scroll-view标签实现自动滑动到底部功能的实例代码
2018/11/09 Javascript
JavaScript Math对象和调试程序的方法分析
2019/05/13 Javascript
[02:24]DOTA2痛苦女王 英雄基础教程
2013/11/26 DOTA
[50:29]2014 DOTA2华西杯精英邀请赛 5 24 DK VS iG
2014/05/26 DOTA
pymssql数据库操作MSSQL2005实例分析
2015/05/25 Python
python bottle框架支持jquery ajax的RESTful风格的PUT和DELETE方法
2017/05/24 Python
python微信跳一跳系列之棋子定位颜色识别
2018/02/26 Python
Python 循环语句之 while,for语句详解
2018/04/23 Python
python读取一个目录下所有txt里面的内容方法
2018/06/23 Python
Python OpenCV图像指定区域裁剪的实现
2019/10/30 Python
Python for循环及基础用法详解
2019/11/08 Python
python 使用cx-freeze打包程序的实现
2020/03/14 Python
Python如何安装第三方模块
2020/05/28 Python
雷蛇美国官网:Razer
2020/04/03 全球购物
新闻学专业应届生求职信
2013/11/08 职场文书
淘宝好评语大全
2014/05/05 职场文书
人民的好儿女观后感
2015/06/18 职场文书
运动员入场前导词
2015/07/20 职场文书
文艺有韵味的诗句(生命类、亲情类...)
2019/07/11 职场文书
mysql配置SSL证书登录的实现
2021/09/04 MySQL