深入理解this绑定原理

前言

前段时间,看了《你不知道的JavaScript》中this的全面解析,讲的特别好,还没看过的小伙伴抓紧去学习,在这边特意整理总结了一波,一起分享学习。在这之前可以先理解一下关于this、call、applay和bind这篇文章。

调用栈

在了解this绑定原理之前,首页要先了解JavaScript的执行栈。(执行栈也叫做调用栈)如果对调用栈还不了解,可以先详细了解深入浅出JavaScript执行上下文和执行栈

JavaScript是单线程的,所有这决定了同一时间只能做一件事情,其他的活动或事情只能排队等候了,于是就生成出一个等候队列的执行栈(Execution Stack)。简单来说,执行栈就是为了到达当前执行位置所调用的所有函数。而调用位置就在当前正在执行的函数的前一个调用中。

function baz() {
    // 当前执行栈是:baz
    // 调用位置是baz前一个调用中,因此是全局作用域
    console.log('baz');
    bar();
}

function bar() {
    // 当前执行栈是:bar
    // 调用位置是bar前一个调用中,因此是baz
    onsole.log('bar');
}

baz()  // baz的调用位置

根据上面的规则,则不难判断出上述代码的调用栈和调用位置。而调用栈和调用位置则决定了this的绑定对象。其绑定规则有四种,只有找到调用位置,才能判断需要应用四条规则中的哪一条。

默认绑定

在非严格模式下,this指向全局对象,严格模式下this则会指向undefined

function foo() {
  console.log(this.a); // this指向全局对象
}
var a = 2;
foo(); // 2
function foo2() {
  "use strict"; // 严格模式this绑定到undefined
  console.log(this.a); 
}
foo2(); // TypeError:a undefined

但是如果在严格模式下调用其他函数,则不影响默认绑定。

function foo() {
  console.log(this.a); // foo函数不是严格模式 默认绑定全局对象
}
var a = 2;
function foo2(){
  "use strict";
  foo(); // 严格模式下调用其他函数,不影响默认绑定
}
foo2();

上述为独立的函数调用,其调位位置为全局作用域,所有this绑定在全局作用域上。

隐式绑定

函数在调用位置,是否有上下文对象,如果有,那么this就会隐式绑定到这个对象上。
也可以简单理解为是否被某个对象包含了。

function foo() {
  console.log(this.a);
}
var a = "Oops, global";
let obj2 = {
  a: 2,
  foo: foo
};
let obj1 = {
  a: 22,
  obj2: obj2
};
obj2.foo(); // 2 this指向调用函数的对象
obj1.obj2.foo(); // 2 this指向最后一层调用函数的对象

// 隐式绑定丢失
let bar = obj2.foo; // bar只是一个函数别名 是obj2.foo的一个引用
bar(); // "Oops, global" - 指向全局

上述代码中函数foo调用位置在obj2上下文中,所有this绑定在obj2作用域中,所以obj2.foo()obj1.obj2.foo()最终都为2,而let bar = obj2.foo实际上就是函数的引用赋给变量bar,调用时,并没有上下文对象,所以会导致隐式绑定丢失。

显式绑定

通过applycallbind将函数中的this强制绑定到指定对象上。

function foo() {
    console.log(this.a);
}
let obj = {
    a: 2
};
foo.call(obj); // 2

需要注意的是:

  • 如果传入了一个原始值(字符串,布尔类型,数字类型),来当做this的绑定对象,这个原始值会转换成它的对象形式。
  • 如果把null或者undefined作为this的绑定对象传入callapplybind,这些值会在调用时被忽略,实际应用的是默认绑定规则。

new绑定

如果对new关键字不太了解,可以先看这篇关于new命令

使用构造调用的时候,this会自动绑定在new期间创建的对象上。

function foo(a) {
  this.a = a; // this绑定到bar上
}
let bar = new foo(2);
console.log(bar.a); // 2

四种绑定规则的优先级

  • 显式绑定 > 隐式绑定 > 默认绑定
  • new绑定 > 隐式绑定 > 默认绑定
function foo() {
    this.a = 100;
}
var obj1 ={
    foo: foo;
}
var obj2 = {}
obj1.foo.call(obj2, 2); // 2  this指向obj2 显式绑定比隐式绑定优先级高。

new obj1.foo(4); // thsi指向new新创建的对象 new绑定比隐式绑定优先级高。

显式绑定和new绑定无法直接比较(会报错),默认绑定是不应用其他规则之后的兜底绑定所以优先级最低。

箭头函数this指向

箭头函数this不会使用这四条绑定规则。

function foo() {
  return (a) => {
    // this继承自foo
    console.log(this.a);
  };
}
let obj1 = {
  a: 2
};
let obj2 = {
  a: 3
};
let bar = foo.call(obj1); // foo this指向obj1
bar.call(obj2); // 输出2 这里执行箭头函数 并试图绑定this指向到obj2

从上述可以得出,箭头函数的this规则:

  • 箭头函数中的this继承于它外面第一个不是箭头函数的函数的this指向。如果没有则指向全局。
  • 箭头函数的 this 一旦绑定了上下文,就不会被任何代码改变。

小结

如果要判断一个运行中函数this绑定,就需要找到这个函数直接调用位置。找到之后就可以顺序应用下面四条规则来判读this绑定对象。

  1. new调用,绑定到新创建对象。
  2. call或者applybind调用,绑定到指定对象。
  3. 由上下文对象调用,绑定到那个上下文对象。
  4. 默认:在严格模式绑定到undefined,否则绑定到全局
    箭头并不适用与上述四条规则,而是由他的外层函数继承而来。

更多优质文章可以访问GitHub博客,欢迎帅哥美女前来Star!!!

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

推荐阅读更多精彩内容

  • 因为全心全意的信任过一个人,把自己的身体和心甚至整个灵魂都交给了对方。 也许是时间不对,更或许是人不对,而最终的结...
    勇敢Planet阅读 156评论 0 1
  • 我有两个爸爸 文/项小小 ▼ ➀ 李东和李南是一对双胞胎。 十几年前的一个晚上,李东和李南两个人一起下海做生意。运...
    项小小阅读 418评论 0 1
  • 先看了书《再见了,可鲁》,再看了电影《导盲犬小Q》。 原来,一个狗和一个人是一样的啊!人与人不一样,狗与狗各不同。...
    经常想你阅读 708评论 0 6
  • 卓越科技APP是“卓越”系列蓝牙电风扇的手机客户端; 使用卓越科技APP通过蓝牙与“卓越”系列蓝牙电风扇建立连接后...
    tthc阅读 168评论 0 0