闭包是JavaScript中函数能记住并访问其定义时词法作用域变量的机制,用于封装私有变量、解决循环事件绑定问题、实现柯里化与高阶函数、模块模式封装,但需防范内存泄漏。

闭包是 JavaScript 中一个函数能记住并访问其定义时所在词法作用域的变量,即使这个函数在别处执行。简单说,就是一个内部函数引用了外部函数的局部变量,并被返回或传递出去,从而让那些变量持续存活。
封装私有变量和方法
JavaScript 没有原生的 private 关键字,但闭包可以模拟私有状态。把变量声明在外部函数里,只通过返回的对象方法暴露有限操作入口。
- 比如计数器:外部无法直接读写 count,只能调用 increment() 或 getCount()
- 避免全局污染,防止命名冲突
- 适合配置管理、单例模块、缓存逻辑等需要内部状态隔离的场景
解决循环绑定事件中的变量捕获问题
用 var 声明的循环变量在异步回调中常出现“全都是最后一个值”的问题,闭包可为每次迭代保存独立副本。
- 典型例子:给多个 绑点击事件,期望点第 3 个就弹出 3,而不是全弹 5
- 方案:用立即执行函数包裹当前 i,或改用 let(块级作用域本质也是闭包思想的简化实现)
实现函数柯里化与高阶函数
柯里化就是把多参数函数拆成一系列单参数函数,每一步都靠闭包保留已传入的参数。
立即学习“Java免费学习笔记(深入)”;
- 例如:add(1)(2)(3) 返回 6,中间每层都靠闭包记住了前面的数值
- 高阶函数如 once()、throttle() 也依赖闭包维持内部开关或时间戳等状态
模块模式与立即执行函数(IIFE)
早期模块化常用 IIFE 创建封闭作用域,返回一个仅暴露必要接口的对象,内部变量完全隐藏。
- 像 (function(){ ... return { init() {}, destroy() {} }; })() 这种写法,就是闭包驱动的模块封装
- 虽然后来有了 ES Module,但在需兼容旧环境或构建轻量工具函数时仍很实用
闭包很强大,但也意味着变量不会被自动回收。长期持有 DOM 引用或大对象时容易引发内存泄漏,用完记得手动清除引用。











