引言:为什么 this 让开发者又爱又恨?
在 JavaScript 的生态系统中,this 是一个极具争议的核心概念。它灵活、动态,却又常常"出人意料"。许多开发者在调试时都会遇到类似问题:
"为什么
this.name是undefined?""明明是对象的方法,怎么
this指向了window?""箭头函数到底有没有
this?"
这些问题的背后,是 JavaScript 独特的执行上下文机制 与动态绑定策略 。与 Java、C++ 等静态绑定语言不同,JavaScript 的 this 是在函数运行时根据调用方式决定的。
本文将带你从零开始,系统性地剖析 this 的四种绑定规则,深入对比普通函数与箭头函数的行为差异,并结合真实开发场景,帮助你彻底掌握 this 的本质。
✅ 本文目标:
- 理解
this的动态绑定机制- 掌握四种绑定规则及其优先级
- 深入理解箭头函数对
this的影响- 解决
this丢失等常见陷阱- 提供最佳实践与调试技巧
第一章:this 的本质------它到底是谁?
1.1 this 不是函数本身,也不是作用域
一个常见的误解是认为 this 指向函数自己,或者指向其词法作用域。但事实并非如此。
            
            
              javascript
              
              
            
          
          function foo() {
  console.log(this === foo);        // false
  console.log(this === window);     // true(非严格模式)
}
foo();- this并不指向函数- foo。
- this也不属于词法作用域(- [[Scope]]),而是属于执行上下文(Execution Context)。
1.2 执行上下文与 this 的关系
每当一个函数被调用时,JavaScript 引擎会创建一个执行上下文,其中包含:
- 变量对象(Variable Object)
- 作用域链(Scope Chain)
- this绑定(This Binding)
this 的值就是在这一时刻,根据函数的调用方式动态确定的。
📌 关键点 :
this在函数定义时无法确定,只有在调用时才能知道它的指向。
第二章:this 的四大绑定规则(核心机制)
JavaScript 中 this 的绑定遵循四种规则,按优先级从高到低排列:
| 优先级 | 规则 | 触发方式 | 
|---|---|---|
| 1 | new绑定 | 使用 new调用构造函数 | 
| 2 | 显式绑定 | 使用 call、apply、bind | 
| 3 | 隐式绑定 | 作为对象方法调用 | 
| 4 | 默认绑定 | 独立函数调用 | 
我们逐个解析。
2.1 默认绑定(Default Binding)------最基础的规则
当函数独立调用时(无上下文对象),this 指向全局对象。
            
            
              javascript
              
              
            
          
          function sayHello() {
  console.log(this);
}
sayHello(); // 浏览器中输出 window严格模式下的变化
在 'use strict' 模式下,独立调用的函数 this 为 undefined。
            
            
              javascript
              
              
            
          
          'use strict';
function strictFunc() {
  console.log(this); // undefined
}
strictFunc();⚠️ 注意 :默认绑定是
this问题的根源之一,尤其是在函数被赋值或作为回调传递时。
2.2 隐式绑定(Implicit Binding)------谁调用,this 就是谁
当函数作为对象的方法被调用时,this 指向该对象。
            
            
              javascript
              
              
            
          
          const user = {
  name: 'Alice',
  greet() {
    console.log(`Hello, I'm ${this.name}`);
  }
};
user.greet(); // Hello, I'm Alice此时 this 指向 user,因为是 user.greet() 的方式调用。
隐式丢失(Implicit Loss)------最常见陷阱
            
            
              javascript
              
              
            
          
          const greetFunc = user.greet;
greetFunc(); // Hello, I'm undefined虽然 greetFunc 引用了 user.greet,但调用时是独立函数调用,因此 this 指向全局或 undefined。
链式调用中的 this
        
            
            
              javascript
              
              
            
          
          const obj = {
  a: {
    name: 'Bob',
    greet() {
      console.log(this.name);
    }
  }
};
obj.a.greet(); // Bob(this 指向 a)this 永远指向直接调用者 ,即最后一个 . 前的对象。
2.3 显式绑定(Explicit Binding)------手动控制 this
JavaScript 提供三种方法显式设置 this:
| 方法 | 语法 | 是否立即调用 | 参数形式 | 
|---|---|---|---|
| call | func.call(obj, a, b) | 是 | 逐个参数 | 
| apply | func.apply(obj, [a,b]) | 是 | 数组 | 
| bind | func.bind(obj, a, b) | 否 | 返回新函数 | 
call 与 apply:立即调用
        
            
            
              javascript
              
              
            
          
          function introduce(age, city) {
  console.log(`${this.name} is ${age} in ${city}`);
}
const person1 = { name: 'Charlie' };
const person2 = { name: 'Diana' };
introduce.call(person1, 25, 'Beijing');   // Charlie is 25 in Beijing
introduce.apply(person2, [30, 'Shanghai']); // Diana is 30 in Shanghaibind:返回绑定 this 的新函数
        
            
            
              javascript
              
              
            
          
          const boundIntroduce = introduce.bind(person1, 25);
boundIntroduce('Hangzhou'); // Charlie is 25 in Hangzhou
// bind 后的函数无法再改变 this
const anotherObj = { name: 'Eve' };
boundIntroduce.call(anotherObj, 'New York'); // 仍输出 Charlie✅ 应用场景:事件监听、定时器回调、函数柯里化。
2.4 new 绑定(New Binding)------构造函数的 this
使用 new 调用函数时,会创建一个新对象,并将 this 指向它。
            
            
              javascript
              
              
            
          
          function Person(name) {
  this.name = name;
  console.log(this); // 新创建的实例
}
const p = new Person('Frank');
console.log(p.name); // Franknew 的执行步骤
- 创建一个空对象({})。
- 将构造函数的 prototype赋给新对象的__proto__。
- 将 this指向新对象。
- 执行构造函数体。
- 若构造函数返回对象,则返回该对象;否则返回新对象。
            
            
              javascript
              
              
            
          
          function BadConstructor() {
  this.value = 1;
  return { value: 2 }; // 显式返回对象
}
new BadConstructor().value; // 2(this 指向返回的对象)第三章:绑定规则的优先级验证
我们通过实验验证四种规则的优先级。
3.1 显式绑定 > 隐式绑定
            
            
              javascript
              
              
            
          
          const obj1 = { name: 'Grace', fn() { console.log(this.name); } };
const obj2 = { name: 'Henry' };
obj1.fn();           // Grace(隐式)
obj1.fn.call(obj2);  // Henry(显式胜出)3.2 new 绑定 > 显式绑定
        
            
            
              javascript
              
              
            
          
          function foo() {
  console.log(this.name);
}
const obj = { name: 'Ivy' };
const boundFoo = foo.bind(obj);
new boundFoo(); // undefined(new 创建新对象,name 未定义)✅ 结论 :
new绑定优先级最高,甚至能"穿透"bind的绑定。
第四章:箭头函数中的 this ------没有自己的 this
4.1 箭头函数的核心特性
ES6 箭头函数(=>)有以下特点:
- 不能作为构造函数(new报错)
- 没有 arguments、super、new.target
- 没有自己的 this,继承外层作用域的this
            
            
              javascript
              
              
            
          
          const arrow = () => {
  console.log(this);
};
arrow(); // 继承外层 this(通常是 window)4.2 箭头函数的 this 继承机制
箭头函数的 this 在定义时确定,而非调用时。
            
            
              javascript
              
              
            
          
          const obj = {
  name: 'Jack',
  regular() {
    console.log(this.name); // Jack
    const inner = () => {
      console.log(this.name); // Jack(继承外层)
    };
    inner();
  },
  arrow: () => {
    console.log(this.name); // undefined(继承全局)
  }
};
obj.regular(); // Jack, Jack
obj.arrow();   // undefined4.3 箭头函数解决 this 丢失
传统方式:
            
            
              javascript
              
              
            
          
          function Timer() {
  this.seconds = 0;
  setInterval(function() {
    this.seconds++; // 错误!
  }.bind(this), 1000);
}箭头函数方式(更简洁):
            
            
              javascript
              
              
            
          
          function Timer() {
  this.seconds = 0;
  setInterval(() => {
    this.seconds++; // 正确!继承 Timer 的 this
  }, 1000);
}4.4 箭头函数的陷阱
            
            
              javascript
              
              
            
          
          const obj = {
  count: 0,
  increment: () => {
    this.count++; // 错误!this 指向全局
  }
};
obj.increment();
console.log(obj.count); // 0✅ 建议 :对象方法不要用箭头函数,除非你明确需要继承外层 this。
第五章:不同环境下的 this 行为
| 环境 | 全局 this | 模块内 this | 
|---|---|---|
| 浏览器 | window | window | 
| Node.js | global | module.exports | 
| 严格模式 | undefined | undefined | 
            
            
              javascript
              
              
            
          
          // Node.js 模块中
console.log(this); // {}
console.log(this === module.exports); // true第六章:常见陷阱与解决方案
6.1 回调函数中的 this 丢失
        
            
            
              javascript
              
              
            
          
          class Counter {
  constructor() {
    this.count = 0;
  }
  start() {
    setInterval(() => {
      this.count++; // 正确(箭头函数)
    }, 1000);
  }
}替代方案(传统):
            
            
              javascript
              
              
            
          
          start() {
  const self = this;
  setInterval(function() {
    self.count++;
  }, 1000);
}6.2 解构导致 this 丢失
        
            
            
              javascript
              
              
            
          
          const { greet } = user;
greet(); // this 丢失✅ 解决方案 :使用 bind() 或保持方法引用。
结语:掌握 this,掌控 JavaScript 的灵魂
this 是 JavaScript 动态特性的集中体现。它既是挑战,也是力量的源泉。通过理解其绑定机制、善用箭头函数、规避常见陷阱,你将能写出更健壮、更可维护的代码。
🔥 记住 :
this不是魔法,而是规则。掌握规则,你就能驾驭它。