我们在写代码的时候,时常会被this弄的傻傻分不清楚,看源码的时候也经常被call啊apply啊弄的头皮发麻。this到底是什么?本文主要根据书上和实际应用做了一些归纳。一般情况下this有4种绑定规则:

1、默认绑定 - this指向全局变量

举例:

       function baz(){
console.log('a',this.a);
};
var a = 2;
baz(); // a 2

在非严格模式下,我们调用了baz,它没有被任何对象包裹,而是暴露在全局环境中,因此此时它被调用的上下文就是全局环境,所以获取到this.a是2;

2、隐式绑定

这种情况应该是最常见的,我们调用的方法被包含在某个对象中,此时应该找到调用这个方法的对象所处的上下文。

举例:

       function baz(){
console.log('a',this.a);
};
var a = 2;
var obj = {
a: 3,
baz: baz
}
obj.baz(); // a, 3

这个时候,this指向调用baz这个方法的对象obj;所以获取的this.a就是obj.a是3;

思考一下下面的this.a的值又是什么:

      function baz(){
console.log('a',this.a);
};
var a = 2;
var obj = {
a: 3,
baz: baz
}; var obj2 = {
a: 4,
baz: obj.baz
}
obj2.baz();

tips: 对象属性的引用链只有最后一层在调用位置中起作用。

3、显式绑定

这里就是函数可以使用的方法call(obj,param1,param2,...)apply(obj,[param1,param2,...]),它们的作用是我们可以显式的改变函数的上下文(this),参数说明:

  • obj:一个对象,将方法的this指向该对象的this;
  • paramX:参数,两者的表现形式不同
       var a = 2;
function baz(){
console.log('a',this.a);
}; var obj = {
a: 3
}; baz.call(obj); // 改变了this,并执行了函数
// a 3

如果直接baz(),则采用的是第一种规则,这里使用baz.call强行将baz的this指向了obj,所以此时baz中的this是obj;

应用场景:带参数的

        var a = 2;

        function baz(p1, p2) {
console.log(this.a, p1, p2);
}; var obj = {
a: 3
}; var foo = function () {
return baz.apply(obj, arguments)
}; foo("hi,", "heimayu"); // 3 hi, heimayu

4、new绑定

在js中,我们经常使用new来对函数进行构造调用,如下:

       function Foo(a) {
this.a = a;
}; var foo = new Foo(3);
console.log("a", foo.a)

这里,使用new Foo()的时候构造出一个新的对象foo并把它绑定到Foo调用中的this上。

5、优先级

在大多数情况下我们找到函数的调用位置,并判断应用哪种规则,就可以快速找到this;结论:

new绑定 > 显式绑定(call,apply) > 隐式绑定 > 全局默认绑定

6、一些例外

规则是死的,程序是活的,总有例外出现。比如啊:

我不关心this是什么

        function foo(a, b) {
console.log("a:" + a, "b:" + b);
}; foo.apply(null, [1, 2]);
// a:1 b:2

ES6箭头函数

箭头函数不使用上面的规则,而是根据 由外层的作用域来决定它的this

       function foo() {
// return function () {
// console.log('第一种情况', this.a)
// }
return (a)=>{
console.log('第二种情况',this.a)
}
}; var obj1 = {
a: 2
}; var obj2 = {
a: 3
}; var bar = foo.call(obj1);
bar.call(obj2);
// 第一种情况:3
// 第二种情况:2

记住箭头函数的绑定无法被修改!!!因此在执行bar的时调用了foo,而foo中的this是指向到obj1的。

最常见的场景:

        var a = 1;
var obj = {
a:2,
foo: function(){
setTimeout(function(){
console.log(this.a)
},100)
}
}; obj.foo();
// 1

在ES5中,我们想打出的值是2,会这样做:

        var a = 1;
var obj = {
a:2,
foo: function(){
let self = this;
setTimeout(function(){
console.log(self.a)
},100)
}
}; obj.foo();

在ES6中,则可以直接这样:

        var a = 1;
var obj = {
a:2,
foo: function(){
setTimeout(()=>{
console.log(this.a)
},100)
}
}; obj.foo();

理解一下:因为箭头函数中的this,指向了foo,foo的上下文是obj。具体的说:箭头函数会继承外层函数调用的this绑定

关于this的全面解析(call,apply,new)的更多相关文章

  1. 完全解析Array.apply(null, { length: 1000 })

    Array.apply(null, { length: 1000 }) 点击打开视频讲解更加详细 在阅读VueJS教程时有这么段demo code: render: function (createE ...

  2. Spring mybatis源码篇章-NodeHandler实现类具体解析保存Dynamic sql节点信息

    前言:通过阅读源码对实现机制进行了解有利于陶冶情操,承接前文Spring mybatis源码篇章-XMLLanguageDriver解析sql包装为SqlSource SqlNode接口类 publi ...

  3. Mybatis源码解析-DynamicSqlSource和RawSqlSource的区别

    XMLLanguageDriver是ibatis的默认解析sql节点帮助类,其中的方法其会调用生成DynamicSqlSource和RawSqlSource这两个帮助类,本文将对此作下简单的简析 应用 ...

  4. Android进阶:七、Retrofit2.0原理解析之最简流程【下】

    紧接上文Android进阶:七.Retrofit2.0原理解析之最简流程[上] 一.请求参数整理 我们定义的接口已经被实现,但是我们还是不知道我们注解的请求方式,参数类型等是如何发起网络请求的呢? 这 ...

  5. Spring源码分析(七)bean标签的解析及注册

    摘要:本文结合<Spring源码深度解析>来分析Spring 5.0.6版本的源代码.若有描述错误之处,欢迎指正. 在上一篇中提到过Spring中的标签包括默认标签和自定义标签两种,而两种 ...

  6. InnoDB recovery过程解析

    本文来自网易云社区. InnoDB如果发生意外宕机了,数据会丢么?对于这个问题,稍微了解一点MySQL知识的人,都会斩钉截铁的回答:不会!为什么?他们也会毫不犹豫的说:因为有重做日志(redo log ...

  7. Mybaits 源码解析 (六)----- 全网最详细:Select 语句的执行过程分析(上篇)(Mapper方法是如何调用到XML中的SQL的?)

    上一篇我们分析了Mapper接口代理类的生成,本篇接着分析是如何调用到XML中的SQL 我们回顾一下MapperMethod 的execute方法 public Object execute(SqlS ...

  8. Gradle--初识

    1.Eclipse从svn导入Gradle项目 1.检出项目的时候不要选新项目,选"做为工作空间中的项目检出",然后点Finish. 2.将项目转为Gradle项目,右键导入的项目 ...

  9. MyBatis 源码分析 - SQL 的执行过程

    * 本文速览 本篇文章较为详细的介绍了 MyBatis 执行 SQL 的过程.该过程本身比较复杂,牵涉到的技术点比较多.包括但不限于 Mapper 接口代理类的生成.接口方法的解析.SQL 语句的解析 ...

  10. MySQL · 引擎特性 · 基于InnoDB的物理复制实现(转载)

    http://mysql.taobao.org/monthly/2016/05/01/ 在开始之前,你需要对InnoDB的事务系统有个基本的认识.如果您不了解,可以参考我之前的几篇关于InnoDB的文 ...

随机推荐

  1. eclipse远程debug服务器上的项目(Tomcat),打开、关闭及常见错误汇总

    我们工作中,有时候测试/生产环境,出现的结果会与我们预计的不一样,只看代码又看不出去问题所在,这个时候就需要远程debug下服务器上的项目. 注意:(1)需要debug的代码,本机代码需与服务器上一致 ...

  2. 【OpenCV3】直线拟合--FitLine()函数详解

    一.FitLine()函数原型 CV_EXPORTS_W void fitLine( InputArray points, // 待输入点集(一般为二维数组或vector点集) OutputArray ...

  3. AFNetworking 3.1.0 使用中某些知识点讲解

    # POST / GET  请求 /*! 首先要知道,POST请求不能被缓存,只有 GET 请求能被缓存.因为从数学的角度来讲,GET 的结果是 幂等 的,就好像字典里的 key 与 value 就是 ...

  4. java连接SqlServer2012

    要用java连接数据库  首先是要通过JDBC驱动 要先去下载一个sqljdbc4.jar,我这里放百度云盘了, 下载地址:链接:http://pan.baidu.com/s/1slJl89B 密码: ...

  5. Python:学习遇到的小问题:记事本写的脚本执行提示SystaxError:(unicode error) 'utf-8'

    学习了一段时间的Python因为懒没有坚持,现在又想学,在用记事本写好py脚本运行时报错:SystaxError:(unicode error) 'utf-8' 解决的方法: 因为我的笔记本系统自带的 ...

  6. Python PIL库学习笔记

    1.PIL简介 Python Imaging Library(缩写为PIL)(在新的版本中被称为Pillow)是Python编程语言的开源库,它增加了对打开,操作和保存许多不同图像文件格式的支持.它适 ...

  7. PKUWC2019退役记

    PKUWC2019 退役记 \(day1\): 进场看T1,发现是个对于所有边的子集求权值和的计数题,以为是个主旋律那样的神仙容斥,完全不会做(退役flag*1).T2是个和虚树有关的计数题,第一个s ...

  8. 1.CentOS6.5下的基础DNS配置

    常规DNS的安全和配置1.安装DNSyum -y install bind bind-utils安装后生成的文件,我们主要配置下面几个/etc/named.conf/var/named/xx这个xx是 ...

  9. mysql使用常见问题

    常见问题之一: 启动mysql时显示: The server quit without updating PID file 1.可能是/usr/local/mysql/data/mysql.pid文件 ...

  10. Q143 重排链表

    给定一个单链表 L:L0→L1→-→Ln-1→Ln , 将其重新排列后变为: L0→Ln→L1→Ln-1→L2→Ln-2→- 你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换. 示例 1: ...