深入理解js中的apply、call、bind
概述
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
深入理解js中的apply、call、bind的更多相关文章
- JS中的apply,call,bind深入理解
在Javascript中,Function是一种对象.Function对象中的this指向决定于函数被调用的方式.使用apply,call 与 bind 均可以改变函数对象中this的指向,在说区别之 ...
- js 中call,apply,bind的区别
call.apply.bind方法的共同点与区别: apply.call.bind 三者都是用来改变函数的this对象的指向: apply.call.bind 三者都可以利用后续参数传参: bind ...
- 图文结合深入理解 JS 中的 this 值
图文结合深入理解 JS 中的 this 值 在 JS 中最常见的莫过于函数了,在函数(方法)中 this 的出现频率特别高,那么 this 到底是什么呢,今天就和大家一起学习总结一下 JS 中的 th ...
- 如何理解js中的this和实际应用中需要避开哪些坑
this是什么 this就是函数内部的关键字 看下面例子理解js中的this // 例子1 function fnOne () { console.log(this) } 'use strict' f ...
- 深入理解Js中的this
深入理解Js中的this JavaScript作用域为静态作用域static scope,但是在Js中的this却是一个例外,this的指向问题就类似于动态作用域,其并不关心函数和作用域是如何声明以及 ...
- 怎么理解js中的事件委托
怎么理解js中的事件委托 时间 2015-01-15 00:59:59 SegmentFault 原文 http://segmentfault.com/blog/sunchengli/119000 ...
- 深入理解JS中的对象(二):new 的工作原理
目录 序言 不同返回值的构造函数 深入 new 调用函数原理 总结 参考 1.序言 在 深入理解JS中的对象(一):原型.原型链和构造函数 中,我们分析了JS中是否一切皆对象以及对象的原型.原型链和构 ...
- 深入理解JS中的对象(三):class 的工作原理
目录 序言 class 是一个特殊的函数 class 的工作原理 class 继承的原型链关系 参考 1.序言 ECMAScript 2015(ES6) 中引入的 JavaScript 类实质上是 J ...
- 如何更好的理解js中的this,分享2段有意思的代码
关于js中this的浅析,大家可以点击[彻底理解js中this的指向,不必硬背]这篇博客了解. 今天遇到2段比较有意思的代码. ----------------第一段----------------- ...
随机推荐
- 作为前端Web开发者,这12个终端命令不可不会
对于开发人员来说,终端是最重要的工具之一.掌握终端,能够有效的提升开发人员的工作流程.使用终端,许多日常任务都被简化为了编写简单的命令并按下 Enter 按钮. 本文列举了一系列 Linux 命令,旨 ...
- JAVAEE企业级应用开发浅谈第一辑
不积跬步无以至千里,不积小流无以成江海 Step1.情景概要 作为一个JAVA WEB 开发人员,在开发web 项目时项目大家都有自己的一些新的体会,对于web 开发出现的一些比较经典的名词大家都会有 ...
- kettle系列一之eclipse开发
1.引言 最近公司开始一个etl项目,底层结合开源的kettle进行开发.那么学习kettle势在必行,kettle的使用在这里就不用介绍了,网上有很多的资料.例如:kettle中文社区,我们在这里主 ...
- SpringMVC 配置
1.在WEB-INF\web.xml中定义前端控制器 <servlet> <servlet-name>springmvc</servlet-name> <se ...
- PHP获取文件扩展名的五种方式
这是我应聘实习时遇到的一道笔试题: 使用五种以上方式获取一个文件的扩展名. 要求:dir/upload.image.jpg,找出 .jpg 或者 jpg , 必须使用PHP自带的处理函数进行处理,方法 ...
- 笔记-JDBC和commons-dbutils
1.前言 玩过Java web的人应该都接触过JDBC,正是有了它,Java程序才能轻松地访问数据库.JDBC很多人都会,但是为什么我还要写它呢?我曾经一度用烂了JDBC,一度认为JDBC不过如此,后 ...
- 使用bower init创建bower.json文件
使用bower init 可以快速创建bower.json文件 bower init 回答一系列问题后就可以了,其中大部分问题可以按enter跳过.
- Caffe Ubuntu16.04 GPU安装
- [Bayesian] “我是bayesian我怕谁”系列 - Continuous Latent Variables
打开prml and mlapp发现这部分目录编排有点小不同,但神奇的是章节序号竟然都为“十二”. prml:pca --> ppca --> fa mlapp:fa --> pca ...
- [转载]ACM搜索算法总结(总结)
原文地址:ACM搜索算法总结(总结)作者:GreenHand 搜索是ACM竞赛中的常见算法,本文的主要内容就是分析它的 特点,以及在实际问题中如何合理的选择搜索方法,提高效率.文章的第一部分首先分析了 ...