概述

js中的apply,call都是为了改变某个函数运行时的上下文环境而存在的,即改变函数内部的this指向。

apply()

apply 方法传入两个参数:一个是作为函数上下文的对象,另外一个是作为函数参数所组成的数组。

var student = {
name : 'xiaoming'
}
function getName(firstName , lastName){
console.log(firstName + ' ' + this.name + ' ' + lastName)
} getName.apply(student , ['MQ' , 'jj']); //MQ xiaoming jj

call()

call 方法第一个参数也是作为函数上下文的对象,但是后面传入的是一个参数列表,而不是单个数组。

var student = {
name : 'xiaoming'
}
function getName(firstName , lastName){
console.log(firstName + ' ' + this.name + ' ' + lastName)
} getName.call(student , 'MQ' , 'jj'); //MQ xiaoming jj

apply(),call()常用方法

1.数组合并

var arr_1 = [1,2,3];
var arr_2 = [4,5,6];
[].push.apply(arr_1,arr_2);
console.log(arr_1) //[1,2,3,4,5,6]

2.获取数组中的最大值和最小值

var num_arr = [3,5,8,1,9];
var max_num = Math.max.apply(Math , num_arr);
var min_num = Math.min.call(Math , 3 , 5 , 8 , 1 , 9)
console.log(max_num) //9
console.log(min_num) //1

3.类(伪)数组使用数组方法

var domNodes = Array.prototype.slice.call(document.getElementsByTagName("*"));

bind()方法

bind()方法与call(),apply()类试,也可改变函数体内的this指向。

bind()方法会创建一个新函数,称为绑定函数,当调用这个绑定函数时,绑定函数会以创建它时传入 bind()方法的第一个参数作为 this,传入 bind() 方法的第二个以及以后的参数加上绑定函数运行时本身的参数按照顺序作为原函数的参数来调用原函数。

var foo = {
bar : 1,
eventBind: function(){
var _this = this;
$('.someClass').on('click',function(event) {
/* Act on the event */
console.log(_this.bar); //1
});
}
}

使用bind解决保存this的问题

var foo = {
bar : 1,
eventBind: function(){
$('.someClass').on('click',function(event) {
/* Act on the event */
console.log(this.bar); //1
}.bind(this));
}
}

 bind参数的使用:

var student = {
name : 'xiaoming'
} function getName(firstName , lastName){
console.log(firstName + ' ' + this.name + ' ' + lastName)
} var getBindName = getName.bind(student,'lihui');
getName('Mrs','jj'); //Mrs jj
getBindName(); //lihui xiaoming undefined
getBindName('jj'); //lihui xiaoming jj
getBindName('Mrs','jj'); //lihui xiaoming Mrs
getName.call(student,'lihui') //lihui xiaoming undefined

 call 是把第二个及以后的参数作为 getName方法的实参传进去,而 getBindName方法的实参实则是在 bind 中参数的基础上再往后排。

apply、call、bind三者之间的比较

var student = {
age : 18
}
var studentInfo = {
getAge () {
return this.age
}
}
console.log(studentInfo.getAge.bind(student)()); //18
console.log(studentInfo.getAge.call(student)); //18
console.log(studentInfo.getAge.apply(student)); //18

 apply/call在改变函数上下文环境之后, 会立即执行函数。而bind()不会立即执行,而是返回一个改变了上下文 this 后的函数。

总结

  • apply 、 call 、bind 三者都是用来改变函数的this对象的指向
  • apply 、 call 、bind 三者第一个参数都是this要指向的对象,也就是想指定的上下文
  • apply 、 call 、bind 三者都可以利用后续参数传参,只是方式不同
  • apply , call改变函数上下文之后会立即调用,而bind会返回一个改变了this的函数,以便以后调用

参考资料

1.【优雅代码】深入浅出 妙用javascript的call , apply , bind

2.javascript中的call , apply详解

  

深入理解js中的apply、call、bind的更多相关文章

  1. JS中的apply,call,bind深入理解

    在Javascript中,Function是一种对象.Function对象中的this指向决定于函数被调用的方式.使用apply,call 与 bind 均可以改变函数对象中this的指向,在说区别之 ...

  2. js 中call,apply,bind的区别

    call.apply.bind方法的共同点与区别: apply.call.bind 三者都是用来改变函数的this对象的指向: apply.call.bind 三者都可以利用后续参数传参: bind ...

  3. 图文结合深入理解 JS 中的 this 值

    图文结合深入理解 JS 中的 this 值 在 JS 中最常见的莫过于函数了,在函数(方法)中 this 的出现频率特别高,那么 this 到底是什么呢,今天就和大家一起学习总结一下 JS 中的 th ...

  4. 如何理解js中的this和实际应用中需要避开哪些坑

    this是什么 this就是函数内部的关键字 看下面例子理解js中的this // 例子1 function fnOne () { console.log(this) } 'use strict' f ...

  5. 深入理解Js中的this

    深入理解Js中的this JavaScript作用域为静态作用域static scope,但是在Js中的this却是一个例外,this的指向问题就类似于动态作用域,其并不关心函数和作用域是如何声明以及 ...

  6. 怎么理解js中的事件委托

    怎么理解js中的事件委托 时间 2015-01-15 00:59:59  SegmentFault 原文  http://segmentfault.com/blog/sunchengli/119000 ...

  7. 深入理解JS中的对象(二):new 的工作原理

    目录 序言 不同返回值的构造函数 深入 new 调用函数原理 总结 参考 1.序言 在 深入理解JS中的对象(一):原型.原型链和构造函数 中,我们分析了JS中是否一切皆对象以及对象的原型.原型链和构 ...

  8. 深入理解JS中的对象(三):class 的工作原理

    目录 序言 class 是一个特殊的函数 class 的工作原理 class 继承的原型链关系 参考 1.序言 ECMAScript 2015(ES6) 中引入的 JavaScript 类实质上是 J ...

  9. 如何更好的理解js中的this,分享2段有意思的代码

    关于js中this的浅析,大家可以点击[彻底理解js中this的指向,不必硬背]这篇博客了解. 今天遇到2段比较有意思的代码. ----------------第一段----------------- ...

随机推荐

  1. 【转载】Retina屏的移动设备如何实现真正1px的线?

    文章转载自 酷勤网 http://www.kuqin.com/ 原文链接:http://www.kuqin.com/shuoit/20150530/346322.html 原文摘要:前些日子总被人问起 ...

  2. 原生js反转字符串

    //直接操作 var str='nama';var rts=str.split('').reverse().join(''); //String上拓展方法String.prototype.revers ...

  3. linux udev、mdev 介绍

    Udev介绍 Udev的下载网址:http://www.kernel.org/pub/linux/utils/kernel/hotplug/udev/ Udev分为三个子计划:namedev,libs ...

  4. table插件

    //动态添加一行function addRow(){ var firstrow=document.getElementById('firstrow'); var firstCopy=firstrow. ...

  5. Android基础知识06—活动的四大启动模式

    ------ 活动的启动模式 ------ 在实际项目中应该根据特定的需求为每个活动指定恰当的启动模式. 四种启动模式: standard . singleTop . singleTask . sin ...

  6. A low-cost wear-leveling algorithm for block-mappingsolid-state disks

    [] Li-Pin Chang,Li-Chun Huang.A low-cost wear-leveling algorithm for block-mapping solid-state disks ...

  7. LINUX 笔记-netstat命令

    netstat命令用于显示与IP.TCP.UDP和ICMP协议相关的统计数据,一般用于检验本机各端口的网络连接情况.netstat是在内核中访问网络及相关信息的程序,它能提供TCP连接,TCP和UDP ...

  8. Java 多线程的一些问题

    1.volatile关键字的作用 一个非常重要的问题,是每个学习.应用多线程的Java程序员都必须掌握的.理解volatile关键字的作用的前提是要理解Java内存模型,volatile关键字的作用主 ...

  9. (转)log4j使用介绍

    原文出自: log4j使用介绍 日志是应用软件中不可缺少的部分,Apache的开源项目Log4j是一个功能强大的日志组件,提供方便的日志记录.以下是个人经验,具体请参考Log4j文档指南. Log4j ...

  10. Nginx监控-Nginx+Telegraf+Influxb+Grafana

    搭建了Nginx集群后,需要继续深入研究的就是日常Nginx监控. Nginx如何监控?相信百度就可以找到:nginx-status 通过Nginx-status,实时获取到Nginx监控数据后,如何 ...