React中的 闭包陷阱

复制代码
import React, { FC, useState, useRef, useEffect } from 'react'

const App: FC = () => {
  // 1. 使用 useState 声明状态
  // React 会在每次重新渲染时,给 `count` 赋予最新的值。
  // 但是,在事件处理函数(如 alertFn)的"闭包"中,捕获的是函数定义时的那个快照。
  const [count, setCount] = useState(0)

  // 2. 使用 useRef 声明一个可变的引用
  // `useRef` 返回一个普通的 JavaScript 对象 { current: 0 }。
  // 这个对象在整个组件的生命周期中是**同一个**对象,不会因为重新渲染而改变引用。
  // 修改 `countRef.current` 不会引起组件重新渲染。
  const countRef = useRef(0)
  
  // 3. 同步副作用
  // 每当 `count` 状态发生变化时,我们手动更新 `countRef.current` 的值。
  // 这样做是为了让 `countRef.current` 始终保持和最新的 `count` 状态同步。
  useEffect(() => {
    countRef.current = count
  }, [count]) // 依赖数组:仅在 count 变化时执行

  // 4. 增加计数的函数
  // 点击按钮会触发状态更新,导致组件重新渲染。
  function add() {
    setCount(count + 1)
  }

  // 5. 弹窗函数(演示闭包陷阱与解决方案)
  function alertFn() {
    // 设置一个 3 秒后的定时器
    setTimeout(() => {
      // ❌ 如果直接使用 `count` (普通变量/快照):
      // 如果你在点击 "alert" 后、3秒等待期间点击了 "add" 按钮,
      // `count` 的值在定时器内部依然是旧的(例如 0),因为它捕获的是点击 "alert" 那一刻的值。
      // 这就是所谓的"闭包陷阱"或"过时的闭包"。

      // ✅ 现在使用 `countRef.current`:
      // `countRef` 对象本身在组件生命周期中是不变的。
      // 虽然闭包捕获的是同一个 `countRef` 对象,但我们访问的是它的 `.current` 属性。
      // 由于 useEffect 的同步作用,`.current` 里存的是最新的 `count` 值。
      // 所以无论你在 3 秒内点击了多少次 add,alert 显示的都是最新的数字。
      alert(`当前最新的 count 值是: ${countRef.current}`)
    }, 3000)
  }

  return (
    <>
      <p>闭包陷阱演示</p>
      <div>
        {/* 显示当前状态 */}
        <span>屏幕显示: {count}</span>
        {/* 触发状态更新 */}
        <button onClick={add}>add</button>
        {/* 触发异步操作 */}
        <button onClick={alertFn}>3秒后 alert 最新值</button>
      </div>
    </>
  )
}

export default App

如何测试:

  1. 点击 add 按钮几次,比如让数字变成 5
  2. 立刻点击 alert 按钮。
  3. 在等待 3 秒的过程中,继续疯狂点击 add 按钮(比如加到 10)。
  4. 结果: 3 秒后弹出的框显示的是 10,而不是你点击 alert 那一刻的 5。这就是 useRef 解决闭包陷阱的威力。
相关推荐
华玥作者4 小时前
[特殊字符] VitePress 对接 Algolia AI 问答(DocSearch + AI Search)完整实战(下)
前端·人工智能·ai
Mr Xu_4 小时前
告别冗长 switch-case:Vue 项目中基于映射表的优雅路由数据匹配方案
前端·javascript·vue.js
前端摸鱼匠4 小时前
Vue 3 的toRefs保持响应性:讲解toRefs在解构响应式对象时的作用
前端·javascript·vue.js·前端框架·ecmascript
sleeppingfrog5 小时前
zebra通过zpl语言实现中文打印(二)
javascript
lang201509285 小时前
JSR-340 :高性能Web开发新标准
java·前端·servlet
好家伙VCC5 小时前
### WebRTC技术:实时通信的革新与实现####webRTC(Web Real-TimeComm
java·前端·python·webrtc
摘星编程6 小时前
React Native鸿蒙版:Image图片占位符
react native·react.js·harmonyos
未来之窗软件服务6 小时前
未来之窗昭和仙君(六十五)Vue与跨地区多部门开发—东方仙盟练气
前端·javascript·vue.js·仙盟创梦ide·东方仙盟·昭和仙君
baidu_247438616 小时前
Android ViewModel定时任务
android·开发语言·javascript
嘿起屁儿整6 小时前
面试点(网络层面)
前端·网络