从“瞎猫碰死耗子”到彻底通透:一个前端开发者的闭包渡劫实录

2 阅读5分钟

一、缘起:一次“反直觉”的Vuex重构

几年前,我在维护一个多版本并行的SaaS项目时,踩到了一个经典的坑:两个不同版本的页面共用同一个Vuex Store,导致修改A版本的数据,B版本也跟着“抽风”。

当时的我,根本不懂什么是闭包,只知道“两个页面不该共享同一份数据”。于是,我凭着直觉写了一个“工厂函数”:

function createVersionStore() {
  return {
    state: { data: {} },
    mutations: { ... }
  };
}

// 为每个版本创建独立的Store实例
const storeV1 = new Vuex.Store(createVersionStore());
const storeV2 = new Vuex.Store(createVersionStore());

当时只觉得“这样就能隔离数据了”,完全没想到,这个“灵光一闪”的操作,竟然暗合了闭包最精髓的设计模式——函数工厂

直到今天,当我彻底搞懂闭包后,才恍然大悟:原来当年那个“瞎猫碰死耗子”的解决方案,正是闭包在工程化开发中的最佳实践。

二、闭包到底是什么?

很多教程会把闭包讲得很玄乎,什么“函数套函数”“作用域链”“垃圾回收”……但对我来说,真正理解闭包,是从三个关键认知突破开始的:

认知突破1:闭包不是“刻意写的”,而是“自然形成的”
闭包的本质,是函数 + 函数被创建时的环境。只要一个函数能访问到它外部作用域的变量,闭包就自动形成了。

最经典的例子:

function outer() {
  let count = 0;
  return function inner() {
    count++;
    console.log(count);
  };
}

const counter = outer();
counter(); // 1
counter(); // 2

这里,inner函数“记住”了outer里的count变量,即使outer已经执行完毕。这就是闭包——函数带着它的“背包”(外部环境)去流浪。

认知突破2:闭包的“生死”由引用决定
闭包之所以能“长生不老”,是因为内部函数持有了外部变量的引用。只要内部函数还活着,外部变量就不会被垃圾回收。

但这也带来了内存泄漏的风险。比如:

function leak() {
  let bigData = new Array(1000000).fill('data');
  return function() {
    console.log('hello'); // 没用到bigData,但bigData仍被闭包持有
  };
}

const fn = leak();
// bigData永远无法被回收,除非fn = null

所以,闭包的“销毁”很简单:断开所有对内部函数的引用,垃圾回收器会自动清理。

认知突破3:闭包不是单例,每次调用都是“新实例”
这是我最容易混淆的点。很多人以为闭包是“全局共享”的,其实不然。

每次调用外部函数,都会创建一套全新的闭包环境。就像工厂生产产品,每次createCounter()都会生成一个独立的计数器:

const c1 = createCounter();
const c2 = createCounter();

c1(); // 1
c2(); // 1(互不干扰!)

这正是我当年Vuex方案的底层原理——用闭包实现数据隔离

三、闭包的三大“陷阱”与破解之道

理解了原理,还要知道闭包在实际开发中的“坑”。

陷阱1:循环中的闭包(var的诅咒)

for (var i = 0; i < 3; i++) {
  setTimeout(() => console.log(i), 1000);
}
// 输出:3, 3, 3(而不是0,1,2

原因:var是函数作用域,所有定时器共享同一个i。循环结束时i=3,所以都输出3。

解决方案:用let(块级作用域)或立即执行函数(IIFE):

for (let i = 0; i < 3; i++) {
  setTimeout(() => console.log(i), 1000);
}
// 输出:0, 1, 2

陷阱2:React Hooks的“闭包陷阱”
在React中,闭包会“记住”组件渲染时的状态快照:

function Counter() {
  const [count, setCount] = useState(0);

  useEffect(() => {
    const timer = setInterval(() => {
      console.log(count); // 永远是0!
      setCount(count + 1);
    }, 1000);
  }, []); // 依赖为空,只执行一次
}

原因:useEffect只在首次渲染时执行,闭包捕获的是count=0的快照。后续状态更新不会重新创建闭包。

解决方案:用函数式更新setCount(c => c + 1),或把count加入依赖数组。

陷阱3:this指向丢失
闭包不保存this,只保存词法作用域:

const obj = {
  name: '张三',
  getName: function() {
    return function() {
      console.log(this.name); // undefined(this指向window)
    };
  }
};

obj.getName()();

解决方案:用箭头函数或that = this保存上下文。

四、闭包的终极真相:环境引用 vs 值快照

这是我最深的认知突破。

很多人(包括曾经的我)以为闭包捕获的是“变量的副本”,其实不完全对。

真相是:闭包捕获的是“环境的引用”,但对基本类型表现为“值快照”,对引用类型表现为“共享数据”。

  • 基本类型(数字、字符串) :闭包像是拍了一张“快照”,后续外部变化不影响闭包内的值。
  • 引用类型(对象、数组) :闭包持有的是“指针”,修改对象会影响所有持有该引用的闭包。

但关键在于:如果对象是在外部函数内部创建的,每次调用都会生成新对象,闭包之间依然隔离。

这正是我当年Vuex方案的精髓——用函数工厂生成独立的数据空间

五、闭包的工程化价值:从理论到实践

闭包不是面试造火箭的玩具,而是解决实际问题的利器:

  • 数据私有化:模拟私有变量,防止外部污染。
  • 模块化开发:Vuex/Pinia的Store工厂、Webpack的模块系统,底层都是闭包。
  • 函数柯里化:动态生成定制函数。
  • 事件处理与回调:保存上下文状态。

最让我感慨的是,当年那个“不懂闭包”的我,凭着“不想写重复代码”的直觉,竟然写出了符合行业标准的解决方案。这说明,好的工程直觉,往往比死记硬背理论更重要

六、结语:闭包不是魔法,是思维工具

闭包不是什么神秘的魔法,它只是JavaScript函数作用域的自然结果。理解闭包,不是为了应付面试,而是为了写出更健壮、更可维护的代码。

从“瞎猫碰死耗子”到“彻底通透”,我的闭包之旅告诉我:真正的掌握,不是记住定义,而是能在实际问题中识别并运用它

希望我的经历,能帮你少走一些弯路。毕竟,闭包这东西,一旦通了,就再也回不去了。


互动话题:你在项目中用过闭包解决过什么实际问题?欢迎在评论区分享!

参考资料:MDN闭包文档、Vue/Pinia源码、JavaScript高级程序设计

作者:一个从Vuex工厂函数悟出闭包真谛的前端开发者