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 解决闭包陷阱的威力。
相关推荐
We་ct35 分钟前
LeetCode 72. 编辑距离:动态规划经典题解
前端·算法·leetcode·typescript·动态规划
小呆呆66644 分钟前
Codex 穷鬼大救星
前端·人工智能·后端
当时只道寻常1 小时前
Vue3 + IntersectionObserver 实现高性能图片懒加载
前端
用户617517157011 小时前
关于普通函数和箭头函数的this
javascript
sakiko_2 小时前
UIKit学习笔记3-布局、滚动视图、隐藏或显示视图
前端·笔记·学习·objective-c·swift·uikit
RPGMZ2 小时前
RPGMakerMZ 地图存档点制作 标题继续游戏直接读取存档
开发语言·javascript·游戏·游戏引擎·rpgmz·rpgmakermz
有一个好名字3 小时前
Agent Loop —— 一切从那个 while 循环开始
前端·javascript·chrome
一天睡25小时3 小时前
Claude Code 指令入门教程
前端
EF@蛐蛐堂3 小时前
【js】浏览器滚动条优化组件OverlayScrollbars
开发语言·javascript·ecmascript
yingyima3 小时前
正则表达式实战:从日志中精准提取关键字段
前端