直接看this

直接看call和bind

首先放一道题:

var a={
a:'haha',
getA: function(){
console.log(this.a);
}
}
var b= {
a:'hello'
}
var getA = a.getA;
var getA2 = getA.bind(a);
function run(fn){
fn();
}
a.getA();//
getA();//
run(a.getA);//
getA2.call(b);//

输出是什么?

可以花几分钟先自己想想。

嘿嘿~

嘿嘿~

嘿嘿~

嘿嘿~

嘿嘿~

嘿嘿~

嘿嘿~

嘿嘿~

嘿嘿~

嘿嘿~

公布答案:

a.getA();//haha
getA();//object
run(a.getA);//object
getA2.call(b);//haha

答对了么?

这里考察了三个点:

形参实参的理解、this的指向、call和bind对this指向的影响。

第一个我相信大家都没问题,这里主要分析后面的两个问题。

this指向问题

this是大部分刚入前端的人都会遇到的坑。

这里通过借鉴网上的文章和自己的理解来总结一下。

this的指向在函数定义的时候是确定不了的,只有函数执行的时候才能确定this到底指向谁,实际上this的最终指向的是那个调用它的对象。

我的理解是,看执行的时候谁在调用就指向谁,但这样理解this也不算完全准确。

这里有三种简单情况:

  1. 如果函数中的this没有调用它的对象,那么this指向的就是window(严格模式下这种情况的this会为空,即undefined)。
  2. 如果函数中的this被不包含子对象的对象所调用,那么this指向的就是调用它的对象。
  3. 如果函数中的this被包含多级对象的对象调用,this指向的也只是它上一级的对象,如下例。
var demoObj = {
a:1,
b:{
fun:function(){
console.log(this.a);
}
}
}
demoObj.b.fun();//undefined

这里this不是指向demoObj对象,而是指向demoObj.b对象,这里找不到demoObj.b对象里的a,所以会输出undefined。

还有三种特殊情况:

  • 还是上面的例子,改一下调用函数的方式,如下。
var demoObj = {
a:1,
b:{
fun:function(){
console.log(this.a);
}
}
}
var newFun = demoObj.b.fun;
newFun();//undefined

这里还是得到undefined,但是this的指向却是window,这里的undefined是因为没找到window对象里的a,才输出的undefined。

虽然函数fun是被对象b所调用,但是在将fun赋值给变量newFun的时候并没有执行,newFun的上级对象window,所以最终执行时指向的是window。

  • 构造函数用new实例对象时对this的影响。
function Fun(){
this.name = "haha";
}
var stu = new Fun();
console.log(stu.name); //haha

这里之所以对象stu.name可以输出haha,是因为new关键字就是创建一个对象实例,这个stu对象中包含了this.name这个属性,相当于复制但却没有执行。在执行时调用这个函数Fun的是对象stu,所以this指向的就是对象stu。

用new操作符创建对象时发生的事情:

第一步: 创建一个Object对象实例。

第二步: 将构造函数的作用域赋给新对象(因此this就指向了这个新对象)。

第三步: 执行构造函数中的代码(这里的执行并不是真的让this指向哪里,而是为这个新对象添加属性)。

第四步: 返回新生成的对象实例

原本的构造函数是window对象的方法,如果不用new操作符而直接调用,那么构造函数的执行对象就是window,即this指向了window。现在用new操作符后,this就指向了新生成的对象。理解这一步至关重要。

  • 有return的函数在new时对this的影响(正常的构造函数是没有return语句),我们先看下面的几个例子。
//例1
function Fun()
{
this.name = 'haha';
return {};
}
var stu = new Fun();
console.log(stu.name); //undefined
//例2
function Fun()
{
this.name = 'haha';
return function(){};
}
var stu = new Fun();
console.log(stu.name); //undefined
//例3
function Fun()
{
this.name = 'haha';
return 123;
}
var stu = new Fun();
console.log(stu.name); //haha
//例4
function Fun()
{
this.name = 'haha';
return undefined;
}
var stu = new Fun();
console.log(stu.name); //haha

可以看出:

如果return的是一个对象,那么this会指向返回的对象,如果return的不是一个对象,那么this还是指向函数的实例。

但是return的是null时比较特殊。虽然null也是对象,但是this还是指向函数的实例。

//例5
function Fun()
{
this.name = 'haha';
return null;
}
var stu = new Fun();
console.log(stu.name); //haha

至此,this就说这么多,相信前三个输出大家都能理解了,接下来说说call和bind。

call和bind

call和apply只有参数不同,这里就只讨论call,因为call和bind参数使用方法是一样的。

  • call是动态的改变this的指向,即换个对象执行原对象方法的方法,并立即执行
  • bind是静态改变this的指向,并返回一个修改后的函数

就拿开始的题目最后一个输出来说:

如果只是使用call的话:

getA.call(b);//hello
getA.call(a);//haha

在执行到这两句时动态改变了this的指向,所以call(b)的输出hello,call(a)的输出haha。

接下来看有bind影响的:

var getA2 = getA.bind(a);

这里getA其实是a.getA大家应该理解,那么getA.bind(a)将this指向a,其实还是返回了a.getA函数赋值给了getA2。注意:其实函数没有变化,但是内部已经将this指向了a

getA2.call(b);//haha
//相当于a.getA.call(b);

之后call在动态调用时,内部的this已经指向了a,不会再指向b,因此会输出haha。

如果还不是很理解,就将绑定a改成b,如下:

var getA2 = getA.bind(b);
getA2.call(a);//hello

此时无论call里是a还是b,都会输出hello,因为内部的this已经被bind绑定指向b了。

总的来说,call方法是在调用时改变this并立即执行这个函数,bind方法可以先改变函数中的this,之后对应的函数可以在需要的时候再调用。


解决了题,再学一点点。

bind的参数可以在执行的时候再次添加,但是要注意的是,参数需要按照形参的顺序添加。例如:

var demoObj = {
name:"haha",
fun:function(a,b,c){
console.log(a,b,c);
}
}
var newFun = demoObj.fun;
var newFun2 = newFun.bind(demoObj,5);
newFun2(7,9);//5,7,9

看看应该就知道怎么用了。

这里放一个很棒的文章JavaScript 的 this 指向问题深度解析

好了,谢谢大家阅读~

深入理解this,bind、call的更多相关文章

  1. 理解Underscore中的_.bind函数

    最近一直忙于实习以及毕业设计的事情,所以上周阅读源码之后本周就一直没有进展.今天在写完开题报告之后又抽空看了一眼Underscore源码,发现上次没有看明白的一个函数忽然就豁然开朗了,于是赶紧写下了这 ...

  2. js学习进阶中-bind()方法

    有次面试遇到的,也是没说清楚具体的作用,感觉自己现在还是没有深刻的理解! bind():绑定事件类型和处理函数到DOM element(父元素上) live():绑定事件到根节点上,(document ...

  3. javascript 中的 bind (编辑中。。。。)

    这篇文章说的非常好!http://my.oschina.net/blogshi/blog/265415 我的体会就是,函数中的this,指的是运行时,它是被哪个对象调用的.因为javascrpit的函 ...

  4. 关于原生js中bind函数的实现

    今天继续研究了bind函数的实现,也知道了shim和polyfill的说法,现在总结一下, if (!Function.prototype.bind) { Function.prototype.bin ...

  5. Boost::bind使用详解

    1.Boost::bind 在STL中,我们经常需要使用bind1st,bind2st函数绑定器和fun_ptr,mem_fun等函数适配器,这些函数绑定器和函数适配器使用起来比较麻烦,需要根据是全局 ...

  6. 依赖注入和Guice理解

    理解依赖注入,这篇文章写得非常好,结合spring的依赖注入分析的. http://blog.csdn.net/taijianyu/article/details/2338311/ 大体的意思是: 有 ...

  7. js中自己实现bind函数的方式

    最近由于工作比较忙,好久都没时间静下心来研究一些东西了.今天在研究 call 和 apply 的区别的时候,看到 github 上面的一篇文章,看完以后,感觉启发很大. 文章链接为 https://g ...

  8. 初学bind

    其实项目中还没有用到. 但自己还是想逐步了解一些高级的JS语法,不是为了炫技,也不像找前端的工作. 主要目的是:1.学习设计思想,提升解决问题的能力2.让自己的脑子动起来,别太笨. 简单的几句话总结一 ...

  9. Computation expressions and wrapper types

    原文地址:http://fsharpforfunandprofit.com/posts/computation-expressions-wrapper-types/ 在上一篇中,我们介绍了“maybe ...

  10. 论JavaScript的作用域

    一直以来本人认为想深入了解一门语言,不光是让自己变成撸sir,更需要时间的锤炼.能经得起时间考验的东西更值得拥有.学习和使用Javascript一晃都7年了,最近才感觉自己对他才有顿悟,不知道是否来得 ...

随机推荐

  1. .NET C# 创建WebService服务简单的例子

    Web service是一个基于可编程的web的应用程序,用于开发分布式的互操作的应用程序,也是一种web服务 WebService的特性有以下几点: 1.使用XML(标准通用标记语言)来作为数据交互 ...

  2. Elasticsearch和HDFS 容错机制 备忘

    1.Elasticsearch 横向扩容以及容错机制http://www.bubuko.com/infodetail-2499254.html 2.HDFS容错机制详解https://www.cnbl ...

  3. 详解 Java 中的三种代理模式

    代理模式 代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式;即通过代理对象访问目标对象.这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能. 这里使用 ...

  4. canvas给图片加水印

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  5. 【译】异步JavaScript的演变史:从回调到Promises再到Async/Await

    我最喜欢的网站之一是BerkshireHathaway.com--它简单,有效,并且自1997年推出以来一直正常运行.更值得注意的是,在过去的20年中,这个网站很有可能从未出现过错误.为什么?因为它都 ...

  6. Kudu原理-kudu的底层数据模型

    Kudu自身的架构,部分借鉴了Bigtable/HBase/Spanner的设计思想.论文的作者列表中,有几位是HBase社区的Committer/PBC成员,因此,在论文中也能很深刻的感受到HBas ...

  7. sparkStreaming序列化问题

    执行sparkSTreaming+kafka 报错如下: org.apache.spark.SparkException: Task not serializable ...... Caused by ...

  8. websocket/dwebsocket 实现前后端的实时通信

    1.  用bottle框架,自己写一个服务端实现: 转载   :http://www.linuxyw.com/813.html 功能:用websocket技术,在运维工具的浏览器上实时显示远程服务器上 ...

  9. HDU 6119 小小粉丝度度熊 (区间去重)【尺取】

    <题目链接> 度度熊决定每天都在星星小姐的贴吧里面签到. 但是度度熊是一个非常健忘的孩子,总有那么几天,度度熊忘记签到,于是就断掉了他的连续签到. 不过度度熊并不是非常悲伤,因为他有m张补 ...

  10. Codeforces Gym 101291C【优先队列】

    <题目链接> 题目大意: 就是一道纯模拟题,具体模拟过程见代码. 解题分析:要掌握不同优先级的优先队列的设置.下面是对优先队列的使用操作详解: priority_queue<int& ...