JavaScript 中 this 的错误认识、绑定规则、常见问题讲解

开发 前端
相信Javascript中的this会使很多同学在工作学习中产生困惑,笔者经过阅读各种资料及实际工作中的应用,做了以下梳理,主要内容包括长期以来大家对this的错误认识及this的绑定规则,箭头函数、实际工作场景中遇到的问题。

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

[[336197]]

一、两种错误认识

1. 指向自身

this 的第一个错误认识是,很容易把 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.   this.count++; // 记录 foo 被调用次数 
  3. foo.count = 0
  4. window.count = 0
  5. for(let i=0; i<10; i++){ 
  6.   if(i > 5){ 
  7.     foo(i); 
  8.   } 
  9. console.log(foo.count, window.count); // 0 4 

2. 指向函数的作用域

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

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

  • 浏览器:在浏览器环境里是没有问题的,全局声明的函数放在了 window 对象下,foo 函数里面的 this 代指的是 window 对象,在全局环境中并没有声明变量 a,因此在 bar 函数中的 this.a 自然没有定义,输出 undefined。
  • Node.js:在 Node.js 环境下,声明的 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. function bar(){ 
  5.   console.log(this.a); 
  6. foo(); 

二、This 四种绑定规则

1. 默认绑定

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

严格模式下调用:

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

非严格模式下调用:

在浏览器环境下会将 a 绑定到 window.a,以下代码使用 var 声明的变量 a 会输出 1。

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

以下代码使用 let 或 const 声明变量 a 结果会输出 undefined

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

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

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

2. 隐式绑定

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

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

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

隐式绑定的隐患:

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

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

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

3. 显示绑定

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

  • 隐式绑定:在一个对象的内部通过属性间接引用函数,从而把 this 隐式绑定到对象内部属性所指向的函数(例如上例中的对象 parent 的 child 属性引用函数 function child(){})。
  • 显示绑定:需要引用一个对象时进行强制绑定调用,js 有提供 call()、apply() 方法,ES5 中也提供了内置的方法 Function.prototype.bind。

call()、apply() 这两个函数的第一个参数都是设置 this 对象,区别是 apply 传递参数是按照数组传递,call 是一个一个传递。

  1. function fruit(...args){ 
  2.   console.log(this.name, args); 
  3. var apple = { 
  4.   name: '苹果' 
  5. var banana = { 
  6.   name: '香蕉' 
  7. fruit.call(banana, 'a', 'b')  // [ 'a', 'b' ] 
  8. fruit.apply(apple, ['a', 'b']) // [ 'a', 'b' ] 

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

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

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

  1. function fruit(name){ 
  2.   console.log(`${this.name}: ${name}`); 
  3. const obj = { 
  4.   name: '这是水果', 
  5. const arr = ['苹果', '香蕉']; 
  6. arr.forEach(fruit, obj); 
  7. // 这是水果: 苹果 
  8. // 这是水果: 香蕉 

4. new 绑定

new 绑定也可以影响 this 调用,它是一个构造函数,每一次 new 绑定都会创建一个新对象。

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

三、优先级

如果 this 的调用位置同时应用了多种绑定规则,它是有优先级的:new 绑定 -> 显示绑定 -> 隐式绑定 -> 默认绑定。

四、箭头函数

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

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

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

五、This 使用中的几个常见问题

1. 通过函数和原型链模拟类

以下示例,定义函数 Fruit,之后在原型链上定义 info 方法,实例化对象 f1 和定义对象 f2 分别调用 info 方法。

  1. function Fruit(name) { 
  2.   this.name = name; 
  3. Fruit.prototype.info = function() { 
  4.   console.log(this.name); 
  5. const f1 = new Fruit('Apple'); 
  6. f1.info(); 
  7. const f2 = { name: 'Banana' }; 
  8. f2.info = f1.info; 
  9. f2.info() 

输出之后,两次结果是不一样的,原因是 info 方法里的 this 对应的不是定义时的上下文,而是调用时的上下文,根据我们上面讲的几种绑定规则,对应的是隐式绑定规则。

  1. Apple 
  2. Banana 

2. 原型链上使用箭头函数

如果使用构造函数和原型链模拟类,不能在原型链上定义箭头函数,因为箭头函数的里的 this 会继承外层函数调用的 this 绑定。

  1. function Fruit(name) { 
  2.   this.name = name; 
  3. Fruit.prototype.info = () => { 
  4.   console.log(this.name); 
  5. var name = 'Banana' 
  6. const f1 = new Fruit('Apple'); 
  7. f1.info(); 

3. 在事件中的使用

举一个 Node.js 示例,在事件中使用时,当我们的监听器被调用时,如果声明的是普通函数,this 会被指向监听器所绑定的 EventEmitter 实例,如果使用的箭头函数方式 this 不会指向 EventEmitter 实例。

  1. const EventEmitter = require('events'); 
  2. class MyEmitter extends EventEmitter { 
  3.   constructor() { 
  4.     super(); 
  5.     this.name = 'myEmitter'
  6.   } 
  7. const func1 = () => console.log(this.name); 
  8. const func2 = function () { console.log(this.name); }; 
  9. const myEmitter = new MyEmitter(); 
  10. myEmitter.on('event', func1); // undefined 
  11. myEmitter.on('event', func2); // myEmitter 
  12. myEmitter.emit('event'); 

 

责任编辑:赵宁宁 来源: Nodejs技术栈
相关推荐

2020-02-19 14:02:49

JavaScriptthis前端

2010-01-15 10:26:08

2019-10-30 14:58:45

MVCAndroid表现层

2013-04-07 10:17:54

WindowsPhon

2011-07-04 08:51:27

编程

2015-06-11 10:33:58

企业级云计算混合云应用

2022-07-31 23:54:24

Linux操作系统

2024-07-01 08:23:20

2009-09-07 16:44:28

Linq String

2011-07-14 14:15:40

ThreadLocal

2015-12-21 11:45:27

C语言常见问题错误

2011-02-22 14:00:16

vsftpd

2019-10-10 15:57:09

云安全混合云架构

2011-04-01 13:55:24

Java

2013-11-14 15:47:29

SDN问题答疑

2011-05-06 15:39:55

硒鼓

2010-09-27 13:45:38

2010-07-21 09:10:02

Perl常见问题

2011-07-21 11:19:51

JAVA

2010-04-23 09:58:30

Oracle管理
点赞
收藏

51CTO技术栈公众号