彻底搞懂javascript中的this的指向

一、前言

我们知道this关键字是一个非常重要的语法。毫不夸张的说不理解它的含义,大部分开发任务,都无法完成。在编写js程序的时候也会经常见到或者用到它,但是有一部分开发朋友对this可能一知半解,下面我们一起来探讨javascript中this的具体含义。
本文分为四部分:

  1. 函数的声明和执行
  2. this(这里)是哪里
  3. 变化多端的this
  4. 亘古不变的this

二、函数的声明和执行

回顾函数的声明和执行,因为一般 this 都是在函数中进行使用

  1. 函数的声明:
    • 使用function关键字声明:function fn(){}
    • 使用表达式赋值声明:var fn = function(){}
  2. 函数的执行:
    • 作为函数调用(默认调用):函数名()
    • 作为对象的方法(通过上下文调用):对象 . 方法名()
    • 作为构造函数(通过new调用):new 函数名()
    • 通过call或apply或bind方法间接调用():函数名 . call()

这里需要强调的不是函数的声明,而是函数的执行,并且需要着重关注位置
函数的执行位置 和 this的所在位置,如图:

函数的声明和执行

三、this是哪里

this在英文中的含义是【这】。那么【这】是【哪】?

this关键字一般存在于函数中,在函数中,我们可能会有一些错误理解:

【这】就是指函数,或者【这】就是函数的作用域

其实,不对!

【这】指的是函数的【执行】上下文

注意【执行】这两个字,如果函数没有执行,那么this是没有内容的,只有当函数执行时,this才被 绑定 了内容。

总结一句话:谁【执行】了this所在的函数,this就是谁。

四、变化多端的this

自然语言中有一些神奇的字眼,在不同的场景下,所表示的意思都有所不同,比如:“开”,这个字:

站在一扇门前,说:“开”。表示开门。
站在一个骰盅前,说:“开”。表示开盅。
站在水龙头前,说:“开”。表示开水龙头。
无间道中,梁朝伟用枪指着刘德华,华仔说:“开”。表示开枪。

程序的this有异曲同工之处,根据不同的场景,this不同的位置,函数不同的执行方式,this所指向的内容都不一样。这里结合常见方式,总结出以下四种形式:

4.1 默认绑定

当一个没有明确隶属对象的函数,被直接调用时。该函数内部的this指向window。

function fn(){
    console.log(this.a);
}
var a = 10;
fn();    //10
// 这里的this指全局对象window

当然要注意,在ES5的严格模式下,没有明确隶属对象的函数在默认执行时,其内部的this指向undefined.

4.2 隐式绑定

所谓隐式绑定,就是将没有明确隶属对象的函数,归属到某个对象,通过该对象执行函数。
此时函数内部的this指向该对象。

function fn(){
    console.log(this.a);
}

var obj = {
    a:10,
    fn:fn
}
obj.fn();    // 10
// 这里的this指obj

隐式绑定会遇到隐式丢失的情况:

  1. 当对象的方法被变量引用时,如果该变量没有从属对象,通过该变量执行函数,那么this会丢失,捕获到window。
  2. 当对象的方法,作为回调函数,传入另一个函数内执行时,this会丢失,捕获到window。
function fn(){
    console.log(this.a);
}
var a = 20;

var obj = {
    a:10,
    fn:fn
}
obj.fn();    // 10

// 隐式丢失:虽然 f 是 obj.fn 的引用,但是 f 的执行,并没有归属对象
var f = obj.fn;
f();    // 20
setTimeout(obj.fn, 100);    // 20

隐式丢失,不好,也好,关键看如何应用。

但,隐式丢失是可以被修复的,这就要使用下一种绑定方式:显示绑定

4.3 显示绑定

所谓显示绑定,就是使用函数的方法,如:callapplybind等,可以强制改变this的指向,如果对函数的方法有疑问,可点击call和apply的使用-基础篇call和apply的使用-扩展篇,稍作学习,本篇不做赘述。

此处以call方法举例:

function fn(){
    console.log(this.a);
}
var a = 20;

var obj = {
    a:10
};

fn.call(obj);    // 10
// 这里的this指obj

可以利用显示绑定的方式,修复隐式丢失问题:

function fn(){
    console.log(this.a);
}
var a = 20;

var obj = {
    a:10
};

// 隐式丢失被解决
fn.call(obj);    // 10

setTimeout(fn.bind(obj),100);    // 10

注意:通过修复隐式绑定,我们发现,显示绑定 的优先级要高于 隐式绑定。

4.4 构造函数绑定

构造函数绑定,又叫new绑定,主要用于面向对象编程。

这里还需要掌握new关键字的原理:

  1. 创建一个新对象
  2. 将函数中的this指向这个新对象
  3. 将这个新对象的__proto__指向函数的prototype
  4. 检查函数中是否主动返回对象,如果没有,则返回前三步处理好的对象
function fn(){
    this.a = 10;
}

var f = new fn();

console.log(f.a);    // 10
// 这里的this指创建出来的对象f

其实,只需要记住,凡是被new执行的函数,默认情况下,其内部的this都被new强行指向new出来的对象,也叫实例。

五、亘古不变的this

其实,不管this的指向如何千变万化,但基本都离不开以上几种情况:

  1. 默认绑定 -----------------------------------> 严格模式绑定到undefined,否则为全局对象window
  2. 隐式绑定(在上下文中调用) --------------> 绑定到当前函数的执行上下文对象
  3. 显示绑定(由call或apply或bind调用) ----> 绑定到指定的对象
  4. 构造函数绑定(由new调用) ---------------> 绑定到新创建的对象

只要掌握并熟练运用以上场景,那么this就会被 “玩弄于股掌之间” ~~~


以上,如有纰漏或不足之处,欢迎留言补充...

共同进步,加油!

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