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 解决闭包陷阱的威力。
相关推荐
wearegogog12320 小时前
基于 MATLAB 的卡尔曼滤波器实现,用于消除噪声并估算信号
前端·算法·matlab
Drawing stars21 小时前
JAVA后端 前端 大模型应用 学习路线
java·前端·学习
品克缤21 小时前
Element UI MessageBox 增加第三个按钮(DOM Hack 方案)
前端·javascript·vue.js
小二·21 小时前
Python Web 开发进阶实战:性能压测与调优 —— Locust + Prometheus + Grafana 构建高并发可观测系统
前端·python·prometheus
小沐°21 小时前
vue-设置不同环境的打包和运行
前端·javascript·vue.js
qq_419854051 天前
CSS动效
前端·javascript·css
烛阴1 天前
3D字体TextGeometry
前端·webgl·three.js
桜吹雪1 天前
markstream-vue实战踩坑笔记
前端
南村群童欺我老无力.1 天前
Flutter应用鸿蒙迁移实战:性能优化与渐进式迁移指南
javascript·flutter·ci/cd·华为·性能优化·typescript·harmonyos
C_心欲无痕1 天前
nginx - 实现域名跳转的几种方式
运维·前端·nginx