不可不知的JavaScript this 指向

JavaScript 中的 this 指向算是 比较令人头疼的问题之一了。this 是一个指向当前执行上下文的关键字,在不同的上下文中指向不同的对象,具体取决于函数被调用的方式。

下面, 让我们花费 10 分钟的时间来一次 this 指向的旅程。

this 指向原则

首先,我们需要明确下 this 的指向原则:

1. 如果函数被当做方法来调用,那它的 this 就是调用它的对象。

var a = 0;
let obj = {
  a: 1,
  fun() {
    console.log(this.a);
  }
};
obj.fun(); // 1
obj.fun.call(window); // 0

2. 如果函数(非箭头函数)被当做函数来调用,那么 this 值要么是全局对象(非严格模式),要么是 undefined(严格模式)。

非严格模式:

var a = 0;
function fun() {
  var a = 1;
  function foo() {
    console.log(this.a);
  }
  foo();
}

fun(); // 0

严格模式:

'use strict';
var a = 0;
function fun() {
  var a = 1;
  function foo() {
    console.log(this.a);
  }
  foo();
}

fun(); // 报错,因为 this 是 undefined

3. 箭头函数不会创建自己的 this,它的 this 来自于外层作用域。箭头函数不能通过 call、apply、bind 等方法来改变它的 this 指向,但可以通过改变外层作用域的 this 指向来改变自身的 this 指向。

var a = 0;

let obj = {
  a: 1,
  fun: () => {
    console.log(this.a);
  }
};

let obj2 = {
  a: 2
};

obj.fun(); // 0
obj.fun.call(obj2); // 0

4. 构造函数中的 this 指向它的实例对象。

function Person(name) {
  this.name = name;
  this.say = function () {
    console.log(this.name);
  };
}

const jack = new Person('jack');
jack.say(); // jack  解析:say中的this指向调用它的对象, 也就是实例对象jack

几道面试题

OK,上面几个原则已经记住了,那我们开始下面的问题:

1. 题目 1

let a = 1;
const b = 2;

function foo() {
  console.log(this.a);
  console.log(this.b);
}
foo();
console.log(window.a);

答案:

undefined
undefined
undefined

解析:

  • let/const 修饰的变量不会挂载到全局上,因此 window.aundefined
  • 函数中的 this 指向window, 因此 this.a, this.b 相当于window.a, window.b,都是 undefined.

2. 题目 2

var name = 'window';
var person1 = {
  name: 'person1',
  foo1: function () {
    console.log(this.name);
  },
  foo2: () => console.log(this.name),
  foo3: function () {
    return function () {
      console.log(this.name);
    };
  },
  foo4: function () {
    return () => {
      console.log(this.name);
    };
  }
};
var person2 = { name: 'person2' };

person1.foo1();
person1.foo1.call(person2);

person1.foo2();
person1.foo2.call(person2);

person1.foo3()();
person1.foo3.call(person2)();
person1.foo3().call(person2);

person1.foo4()();
person1.foo4.call(person2)();
person1.foo4().call(person2);

答案&解析:

var name = 'window';
var person1 = {
  name: 'person1',
  foo1: function () {
    console.log(this.name);
  },
  foo2: () => console.log(this.name),
  foo3: function () {
    return function () {
      console.log(this.name);
    };
  },
  foo4: function () {
    return () => {
      console.log(this.name);
    };
  }
};
var person2 = { name: 'person2' };

person1.foo1(); // person1   解析:方法中的this指向调用它的那个对象
person1.foo1.call(person2); // person2   解析:call改变this指向为person2

person1.foo2(); // window   解析:箭头函数没有自己的this,它的this来源于外层作用域,也就是window
person1.foo2.call(person2); // window   解析:箭头函数不能通过call来改变this指向

person1.foo3()(); // window   解析:person1.foo3()是一个函数,非严格模式下,函数的this指向window.
person1.foo3.call(person2)(); // window   解析:person1.foo3.call(person2)改变了foo3的this指向,返回仍然是一个函数,this指向widnow
person1.foo3().call(person2); // person2   解析:person1.foo3()是一个函数,call改变了该函数的this指向为person2

person1.foo4()(); // person1   解析:箭头函数没有自己的this,它的this来源于外层作用域,也就是foo4,foo4的this指向调用它的那个对象,也就是person1
person1.foo4.call(person2)(); // person2    解析:call改变了foo4的this指向为person2, 箭头函数的this指向也是person2
person1.foo4().call(person2); // person1    解析:箭头函数不能通过call来改变this指向,它的this来源于外层作用域,也就是person1

3. 题目 3

var name = 'window';
function Person(name) {
  this.name = name;
  this.obj = {
    name: 'obj',
    foo1: function () {
      return function () {
        console.log(this.name);
      };
    },
    foo2: function () {
      return () => {
        console.log(this.name);
      };
    }
  };
}
var person1 = new Person('person1');
var person2 = new Person('person2');

person1.obj.foo1()();
person1.obj.foo1.call(person2)();
person1.obj.foo1().call(person2);

person1.obj.foo2()();
person1.obj.foo2.call(person2)();
person1.obj.foo2().call(person2);

答案&解析:

var name = 'window';
function Person(name) {
  this.name = name;
  this.obj = {
    name: 'obj',
    foo1: function () {
      return function () {
        console.log(this.name);
      };
    },
    foo2: function () {
      return () => {
        console.log(this.name);
      };
    }
  };
}
var person1 = new Person('person1');
var person2 = new Person('person2');

person1.obj.foo1()(); // window    解析:函数的this指向全局,也就是window
person1.obj.foo1.call(person2)(); // window    解析:函数的this指向全局,也就是window
person1.obj.foo1().call(person2); // person2    解析:call改变this的指向为person2, person2.name 是 'person2'

person1.obj.foo2()(); // obj    解析:箭头函数的this来源于外层作用域,也就是foo2,foo2的this指向调用它的那个对象,也就是obj
person1.obj.foo2.call(person2)(); // person2     解析:箭头函数的this来源于外层作用域,也就是foo2。call改变foo2的this指向为person2
person1.obj.foo2().call(person2); // obj    解析:箭头函数的this无法通过call来改变

4. 题目 4

var obj = {
  a: 1,
  foo: function (b) {
    b = b || this.a;
    return function (c) {
      console.log(this.a + b + c);
    };
  }
};
var a = 2;
var obj2 = { a: 3 };

obj.foo(a).call(obj2, 1);
obj.foo.call(obj2)(1);

答案和解析:

  • obj.foo(a).call(obj2, 1)
    obj.foo(a)返回的是一个函数,函数中引用了外层函数的变量 b, 这里是 2。由于形成了闭包,这个 b 会长存在内存中。
function(c) {
  console.log(`${this.a} + ${b} + ${c}`);
}

然后,通过 call 改变该函数的指向为 obj2, 因此 this.a 等同于 obj.a,也就是 2.

因此答案是:3 + 2 + 1

  • obj.foo.call(obj2)(1)
    obj.foo.call(obj2)改变 foo 的 this 指向为 obj2, 且没有传参。因此 b = this.a 也就是 obj2.a。也就是 3。b 长存内存,是 3
    因此返回的函数是:
function(c) {
  console.log(`${this.a} + ${b} + ${c}`);
}

然后指向函数,参数是 1。函数的 this 指向 window。this.a 等同于 window.a,也就是 2

因此答案是:2 + 3 + 1

最后

如果你做完这几道题不过瘾,可以去刷一下这篇博客 再来 40 道 this 面试题酸爽继续(1.2w 字用手整理)

本文中的大多数问题都来自于该博客。40 道面试题都答对,this 指向 so easy。

参考文档:

再来 40 道 this 面试题酸爽继续(1.2w 字用手整理)

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

推荐阅读更多精彩内容