彻底弄懂Javascript中的this

在很长的一段时间之内,我一直以为作用域就是上下文,这也就对JavaScript中的this理解增加了很多麻烦,所以这篇文章开篇第一个要陈诉的概念就是作用域和上下文不是一个概念。作用域(scope) 是指变量的可访问性,上下文是来决定this。(注意执行期上下文指的是作用域,这是JavaScipt规范,所以得遵守)

在JavaScript中只有两种作用域,一种是全局作用域,另一个就是函数作用域。上下文则会与this息息相关,而this是在运行的时候进行绑定的,它的上下文取决于函数在哪里被调用,this的绑定和函数声明的位置没有任何关系。

当一个函数被调用时,会创建一个活动记录(即执行上下文)。这个活动会包含函数在哪里被调用,函数的调用方法,传入的参数信息等信息。this就是记录的其中一个属性,会在函数的执行过程中用到。

以上引用出自《你不知道的JavaScript(上卷)》,在这里强烈推荐这本书,字字珠玑。

再次强调:this实际上是在函数被调用的时候发生绑定,它指向什么完全取决于函数在哪里调用。

调用位置

接下来我们看看函数调用包括哪几种情况,只有正确的知道函数调用的位置,才能正确的明白this的指向问题。

默认绑定(全局调用)

var a = 2;
function foo() {
    console.log(this.a)
}
foo();

以上就是默认绑定,foo函数是直接调用的。

隐式绑定

b = 2;
var obj = {
    b: 3,
    foo: foo
}

function foo () {
    console.log(this.b);
}
obj.foo(); // 3

这里为什么叫做隐式绑定,因为这个foo函数无论是在obj里面声明还是在obj外面声明,他实际上都是不属于obj这个对象的(obj只是记录了foo这个属性的引用值),但是最后在执行的时候this却被绑定到了obj这个对象上下文中。当然如果有多个对象链式调用,this只会绑定到最后一层。obj2.obj1.foo(),this是绑定到obj1这个对象上下文中。

当然这里有一个注意点

var obj = {
    b: 3,
    foo: foo
}
function foo () {
    console.log(this.b);
}
var bar = obj.foo;

bar(); // 2 

这里实际上bar直接是foo的引用,就相当于var bar = obj.foo = foo,我们打印一下可以发现

console.log(bar === foo && foo === obj.foo && bar === obj.foo) // true

所以此时就和第一种默认绑定一样,bar函数是直接在全局上下文中被调用的,所以this会指向全局。

还有一种就是嵌套函数了

b = 2;
var obj = {
    b: 3,
    foo: foo
}
function foo () {
    console.log('foo', this.b);// 3
    foo2();
}
function foo2() {
    console.log('foo2', this.b); // 2
}
obj.foo();
    

实际上foo2也是直接被(window)调用了。

显示绑定call,apply,bind

通过call,apply,bind函数可以强制某个函数在哪个对象(或者上下文)中被调用

b = 2;

var obj = {
    b: 3,
    foo: foo
}

function foo () {
    console.log('foo', this.b);
}

foo.call(obj); // 3

当然如果你传入的是一个基本类型的值,那么JavaScript会把它转换成它的对象形式。

new绑定

说到new操作符,就不得不说它的内部工作原理了,我们在执行new操作的时候究竟执行了什么。

1 创建一个全新的对象 var obj = {}
2 这个新对象的原型会被执行[[原型]]连接 obj[[prototype]] = Fun.prototye
3 这个新对象会绑定到函数调用的this Fun.bind(obj)
4 如果函数没有返回其他对象,那么会返回这个新创建的对象 return obj;

所以new绑定实质还是显式绑定。

总结一下我们可以按照下面的顺序进行判断

1 函数是否在new中调用(new 绑定),如果是this绑定的就是返回的新对象
2 函数是否通过call、apply(显式绑定)如果是this绑定的是那个指定的对象
3 函数是否在某个上下文对象中调用(隐式绑定),如果是,this绑定的是那个上下文无关文法对象
4 如果都不是那么就是默认绑定,this绑定的就是全局对象或者undefined(严格模式)

例外

凡事总有例外,如果你把null、undefined作为this的绑定对象传入call、apply或者bind那么实际上,这些值在执行的时候会被忽略,实际使用的是默认绑定。那么什么情况下我们会去绑定一个null或者undefined的呢?一种就是用apply来展开一个数组,当然这种方法的确很实用(不过在ES6中出现了...操作符来展开数组)。

function foo(a, b) {return a + b}
foo.apply(null, [2, 3]);

箭头函数,箭头函数中的this是根据外层作用域来决定this的,也就是说箭头函数中的this就和箭头函数在哪里声明有关系了。

a = 2;

var obj = {
    a: 3,
    foo: foo
}

function foo () {
    return () => {
        console.log(this.a);
    };  
}


var fun = foo.call(obj);

fun(); // 3 此时箭头函数的外层作用域为foo,foo函数被调用时,this被绑定在了obj对象上
a = 2;

var obj = {
    a: 3,
    foo: foo
}

var arrowFun = () => {
    console.log(this.a);
}

function foo () {
    return arrowFun;    
}

var fun = foo.call(obj);

fun(); //2 箭头函数的外层作用域为全局作用域,全局作用域中的this指向全局上下文

最后欢迎大家关注我的个人博客,将会有更多的精彩文档,喜欢的话也可以给个star。

Github链接

DJL箫氏的博客

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,029评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,395评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,570评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,535评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,650评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,850评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,006评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,747评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,207评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,536评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,683评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,342评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,964评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,772评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,004评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,401评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,566评论 2 349

推荐阅读更多精彩内容

  • 今天在朋友圈发了我的开心与焦虑,其实开心的事情不止是宝贝生日,还有德国签证下来了,还有妈妈说家里后面的坪打好了,宝...
  • 我们遇到一点点事情总是抱怨,消极的情绪对待这件事情,一味的抱怨根本解决不了任何的事情,与其整天活在抱怨中,不...
    wh王辉阅读 78评论 0 0