详解JavaScript中的this

相信javascript中的this会使很多同学在工作学习中产生困惑,笔者也同样是,经过阅读各种资料及实际工作中的应用,做了以下梳理,主要内容包括长期以来大家对this的错误认识及this的绑定规则,箭头函数、实际工作场景中遇到的问题,希望对于有此困惑的你能有所帮助。

@五月君,NodeJs工程师,慕课网认证作者,热爱技术,喜欢分享的90后青年。Github:https://github.com/Q-Angelo/summarize

快速导航

错误认识

指向自身

人们很容易把this理解成指向函数自身,其实this的指向在函数定义阶段是无法确定的,只有函数执行时才能确定this到底指向谁,实际上this的最终指向是调用它的那个对象。

下面示例,声明函数foo,执行foo.count=0时,向函数对象foo添加了一个属性count。但是函数foo内部代码this.count中的this并不是指向那个函数对象,for循环中的foo(i)掉用它的对象是window,等价于window.foo(i),因此函数foo里面的this指向的是window。

  1. function foo(num){
  2. console.log("foo: " + num);
  3. //记录foo被调用次数
  4. this.count++;
  5. }
  6. foo.count = 0;
  7. for(let i=0; i<10; i++){
  8. if(i > 5){
  9. foo(i);
  10. }
  11. }
  12. console.log(foo.count); // 0

指向函数的作用域

对this的第二种误解就是this指向函数的作用域,

以下这段代码,在foo中试图调用bar函数,是否成功调用,取决于环境。

  • window,在chrome console环境里是没有问题的,全局声明的函数放在了window下,foo函数里面的this代指的是window对象,在全局环境中并没有声明变量a,因此在bar函数中的this.a自然没有定义,输出undefined。

  • nodejs,在node环境下,声明的function 不会放在global全局对象下,因此在foo函数里调用this.bar函数会报 TypeError: this.bar is not a function 错误,调用bar函数,要省去前面的this。

  1. function foo(){
  2. var a = 2;
  3. this.bar();
  4. }
  5. function bar(){
  6. console.log(this.a);
  7. }
  8. foo();

this绑定规则

默认绑定

当函数调用属于独立调用(不带函数引用的调用),无法调用其他的绑定规则,我们给它一个称呼“默认绑定”,在非严格模式下绑定到全局对象,在使用了严格模式(use strict)下绑定到undefined。

严格模式下调用

  1. 'use strict'
  2. function demo(){
  3. console.log(this.a); // TypeError: Cannot read property 'a' of undefined
  4. }
  5. const a = 1;
  6. demo();

非严格模式下,在浏览器 window 全局对象下会将 a 绑定到 window.a

如果是 var 声明的以下代码会输出 1

  1. function demo(){
  2. console.log(this.a); // 1
  3. }
  4. var a = 1;
  5. demo();

如果是 let 声明的以下代码会输出 undefined

  1. function demo(){
  2. console.log(this.a); // undefined
  3. }
  4. let a = 1;
  5. demo();

在举例子的时候其实想要重点说明 this 的默认绑定关系的,但是你会发现上面两种代码因为分别使用了 var、let 进行声明导致的结果也是不一样的,归其原因涉及到 顶层对象的概念

Issue: Nodejs-Roadmap/issues/11 里有童鞋提到这个疑问,也是之前的疏忽,再简单聊下顶层对象的概念,顶层对象(浏览器环境指 window、Node.js 环境指 Global)的属性和全局变量属性的赋值是相等价的,使用 var 和 function 声明的是顶层对象的属性,而 let 就属于 ES6 规范了,但是 ES6 规范中 let、const、class 这些声明的全局变量,不再属于顶层对象的属性。

非严格模式下,在 node 环境中,不会将 a 绑定到 global,因此下面输出undefined

  1. function demo(){
  2. console.log(this.a); // undefined
  3. }
  4. let a = 1;
  5. demo();

注意:项目代码中,要么使用严格模式要么使用非严格模式,不要混合使用,也许会给你造成一些意外的bug。

隐式绑定

在函数的调用位置处被某个对象包含,拥有上下文,看以下示例:

  1. function child() {
  2. console.log(this.name);
  3. }
  4. let parent = {
  5. name: 'zhangsan',
  6. child,
  7. }
  8. parent.child(); // zhangsan

函数在调用时会使用parent对象上下文来引用函数child,可以理解为child函数被调用时parent对象拥有或包含它。

隐式绑定的隐患

被隐式绑定的函数,因为一些不小心的操作会丢失绑定对象,此时就会应用最开始讲的绑定规则中的默认绑定,看下面代码:

  1. {
  2. function child() {
  3. console.log(this.name);
  4. }
  5. let parent = {
  6. name: 'zhangsan',
  7. child,
  8. }
  9. let parent2 = parent.child;
  10. var name = 'lisi';
  11. parent2();
  12. }

将parent.child函数本身赋给parent2,调用parent2()其实是一个不带任何修饰的函数调用,因此会应用默认绑定。

显示绑定

显示绑定和隐式绑定从字面意思理解,有一个相反的对比,一个表现的更直接,一个表现的更委婉,下面在看下两个规则各自的含义:

  • 隐式绑定 在一个对象的内部通过属性间接引用函数,从而把this隐式绑定到对象内部属性所指向的函数(例如上例中的对象parent的child属性引用函数function child(){})。

  • 显示绑定 需要引用一个对象时进行强制绑定调用,js有提供call()、apply()方法,ES5中也提供了内置的方法 Function.prototype.bind

call、apply这两个函数的第一个参数都是设置this对象,关于两个个函数的区别可以查看 [函数] call和apply的使用与区别?

  1. // 水果对象
  2. function fruit(){
  3. console.log(this.name, arguments);
  4. }
  5. var apple = {
  6. name: '苹果'
  7. }
  8. var banana = {
  9. name: '香蕉'
  10. }
  11. fruit.call(banana, banana, apple) // 香蕉 { '0': { name: '香蕉' }, '1': { name: '苹果' } }
  12. fruit.apply(apple, [banana, apple]) 苹果 { '0': { name: '香蕉' }, '1': { name: '苹果' } }

下面是bind绑定的示例,只是将一个值绑定到函数的this上,并将绑定好的函数返回,只有在fruit函数才会输出信息,例:

  1. function fruit(){
  2. console.log(this.name);
  3. }
  4. var apple = {
  5. name: '苹果'
  6. }
  7. fruit = fruit.bind(apple);
  8. fruit(); // 苹果

除了以上call、apply、bind还可以通过上下文context,例:

  1. function fruit(name){
  2. console.log(name, this.name);
  3. }
  4. const obj = {
  5. name: '这是水果',
  6. }
  7. const arr = ['apple', 'banana', 'pear'];
  8. arr.forEach(fruit, obj);
  9. // apple 这是水果
  10. // banana 这是水果
  11. // pear 这是水果

new绑定

也是一种可以影响this调用的方法,需要清楚new绑定,它是一个构造函数,每一次new绑定都会初始化新创建的对象。

  1. function fruit(name){
  2. console.log(name, this);
  3. }
  4. const f1 = new fruit('apple'); // apple fruit {}
  5. const f2 = new fruit('banana'); // banana fruit {}
  6. console.log(f1, f2, f1 === f2); // fruit {} fruit {} false

优先级

  • new绑定
  • 显示绑定
  • 隐式绑定
  • 默认绑定(严格模式下会绑定到undefined)

箭头函数

箭头函数并非使用function关键字进行定义,也不会使用上面所讲解的this四种标准规范,箭头函数会继承自外层函数调用的this绑定。

执行 fruit.call(apple)时,箭头函数this已被绑定,无法再次被修改。

  1. function fruit(){
  2. return () => {
  3. console.log(this.name);
  4. }
  5. }
  6. var apple = {
  7. name: '苹果'
  8. }
  9. var banana = {
  10. name: '香蕉'
  11. }
  12. var fruitCall = fruit.call(apple);
  13. fruitCall.call(banana); // 苹果

this在项目中使用问题总结

  • React中使用this注意的问题
  1. class App extends React.Component{
  2. componentDidMount(){
  3. console.log(this);
  4. //注意this在setTimeout函数外代表的是App这个对象
  5. setTimeout(function(){
  6. this.setState({ //这里的this指的是windows对象
  7. initDone:true
  8. })
  9. },1000);*/
  10. //两种解决方案:
  11. //1. var that = this
  12. setTimeout(() => {
  13. that.setState({
  14. initDone:true
  15. })
  16. },1000);
  17. //2.使用es6的箭头函数
  18. setTimeout(() => {
  19. this.setState({
  20. initDone:true
  21. })
  22. },1000);
  23. }
  24. }