Javascript中的闭包

1、什么是闭包

当一个函数在其定义时的作用域之外执行,并且仍然可以访问当时的变量,这个现象就叫做闭包。

javascript 复制代码
function createCounter() {
  let count = 0

  return function () {
    count++
    return count
  }
}

const counter = createCounter()
counter() // 1
counter() // 2
  • createCounter() 执行完后,按理说作用域应该销毁

  • 但返回的函数 仍然引用了 count

  • JS 引擎发现:

    👉 这个作用域不能回收

  • 于是形成了一个 被"保活"的作用域 ------ 这就是闭包

重点:不是函数产生了闭包,而是函数使用了外层作用域的变量,并且在外层作用域之外执行。

2、vue中的闭包案例

javascript 复制代码
export function useUser() {
  const user = ref(null)

  function setUser(data) {
    user.value = data
  }

  return {
    user,
    setUser
  }
}
  • useUser 执行完后,本应销毁

  • setUser 仍然在组件里被调用

  • setUser 引用了 user

  • 所以:useUser 的作用域被保留

👉 组合式 API = 闭包驱动的设计

这不是巧合,是语言能力直接影响框架设计

3、为什么 for + setTimeout 会"翻车"

javascript 复制代码
for (var i = 0; i < 3; i++) {
  setTimeout(() => {
    console.log(i)
  }, 1000)
}
//输出3,3,3
  • var i 只有一个作用域

  • 所有回调 共享同一个 i

  • 回调执行时,循环早就结束了

javascript 复制代码
for (let i = 0; i < 3; i++) {
  setTimeout(() => {
    console.log(i)
  }, 1000)
}

👉let 为每次循环创建了一个新的词法作用域

👉 每个回调闭包住的是 不同的 i

**4、**闭包 ≠ 内存泄漏

闭包会延长变量生命周期,
如果引用链一直存在,才可能导致内存无法释放。

Vue 中真正的问题通常是:

  • 全局事件没解绑

  • 定时器没清理

  • 引用被缓存到单例中

不是"闭包的锅"。

5、用闭包实现"私有变量"

javascript 复制代码
function createUser() {
  let password = '123456'

  return {
    check(pwd) {
      return pwd === password
    }
  }
}

JS 没有 private,但闭包就是私有化方案

**6、**Vue 业务里的真实闭包模式

javascript 复制代码
function useRequest() {
  let loading = false

  async function run(promise) {
    loading = true
    try {
      return await promise
    } finally {
      loading = false
    }
  }

  return { run }
}

👉loading 被 run 闭包住

👉 状态与行为天然绑定

7、思考

  • 为什么 setup 执行一次,但状态可以长期存在?

  • 闭包和模块作用域有什么相同与不同?

  • 如果一个闭包持有一个很大的对象,什么时候会释放?

  • 为什么 hooks / composables 天然适合用闭包实现?

  • 如果让你写一个 useDebounce,闭包在其中起什么作用?

相关推荐
caimouse2 小时前
reactos编码规范
c语言·开发语言
xieliyu.6 小时前
Java算法精讲:双指针(三)
java·开发语言·算法
CryptoPP7 小时前
快速对接东京证券交易所API数据:实战指南与代码示例
开发语言·人工智能·windows·python·信息可视化·区块链
ZC跨境爬虫7 小时前
跟着 MDN 学JavaScript day_7:数学运算与逻辑判断实战测试
开发语言·前端·javascript·学习·ecmascript
凌云拓界7 小时前
文件管理:让AI安全操作你的电脑 ——CogitoAgent开发实战(三)
javascript·人工智能·架构·开源·node.js
凌云拓界8 小时前
联网能力:让AI看见更广阔的世界 ——CogitoAgent开发实战(四)
javascript·人工智能·架构·node.js·创业创新
阳区欠8 小时前
【LangChain】LLM基础介绍
开发语言·python·langchain
Jinkxs8 小时前
Java 跨域14-Java 与区块链(Hyperledger)集成
java·开发语言·区块链
HYCS9 小时前
用pixi.js实现fabric.js(六):从线性代数的角度理解编辑器交互
前端·javascript·canvas
晨曦中的暮雨9 小时前
Golang速通(Javaer版)
java·开发语言·后端·golang