react的useMemo 如何优化?

好的,我们来深入讲讲 React 的 useMemo 以及它的优化思路。我们一步步拆解,确保你能真正理解什么时候用、怎么用,以及可能的坑。


1️⃣ useMemo 的本质

useMemo 是一个 性能优化 Hook ,它的作用是 缓存计算结果 ,避免在每次渲染时重复计算开销大的值

语法:

scss 复制代码
const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]);
  • 第一个参数:返回一个值的函数(这个值会被缓存)
  • 第二个参数:依赖数组,只有依赖发生变化时,函数才会重新执行

💡 核心:useMemo 不是用来缓存组件的渲染 ,而是用来缓存计算结果


2️⃣ 典型使用场景

场景 1:开销大的计算

typescript 复制代码
function MyComponent({ number }) {
  const factorial = useMemo(() => {
    console.log('计算阶乘');
    let result = 1;
    for (let i = 2; i <= number; i++) {
      result *= i;
    }
    return result;
  }, [number]);

  return <div>{factorial}</div>;
}

✅ 优化点:

  • 如果没有 useMemo,每次父组件渲染,阶乘都会重新计算。
  • useMemo 后,只有 number 改变时才计算。

场景 2:避免引用类型变化导致子组件重复渲染

javascript 复制代码
function Parent({ data }) {
  const filteredData = useMemo(() => data.filter(item => item.active), [data]);

  return <Child data={filteredData} />;
}
  • 如果不加 useMemo,每次渲染 Parent 时,filteredData 都是一个新数组,Child 会触发不必要的渲染。
  • useMemo 缓存数组引用,减少 Child 渲染。

3️⃣ 使用 useMemo 的注意点

  1. 不要滥用

    • useMemo 本身也有开销(创建函数、比较依赖)。
    • 小计算、简单对象不要用,可能比不使用更慢。
    • 适合计算复杂、数据量大或依赖变化少的场景。
  2. 依赖数组必须准确

    • 缺失依赖会导致缓存结果不同步
    • 多余依赖会降低性能
  3. 不缓存副作用

    • useMemo 不能替代 useEffect
    • 它只是缓存,不执行副作用(比如 API 请求)

4️⃣ 优化策略总结

  1. 缓存重计算值

    • 复杂计算、大数组操作、大对象计算
    • 示例:阶乘、排序、过滤、聚合计算
  2. 缓存引用类型

    • 避免每次渲染生成新的数组/对象,导致子组件不必要渲染
    • 常配合 React.memouseCallback
  3. 分离组件

    • 如果父组件渲染频繁,且子组件依赖开销大的计算,考虑用 useMemo 或拆成独立组件
  4. 依赖最小化

    • 确保依赖数组只放必须的值,避免频繁失效

⚡ 小技巧

  • 配合 React.memo
javascript 复制代码
const Child = React.memo(({ data }) => {
  console.log('Child 渲染');
  return <div>{data.length}</div>;
});

function Parent({ items }) {
  const activeItems = useMemo(() => items.filter(item => item.active), [items]);
  return <Child data={activeItems} />;
}
  • Child 只在 activeItems 改变时渲染,而不是每次 Parent 渲染。

  • 配合 useCallback 缓存函数

    • 当函数作为 prop 传递时,避免引用变化
ini 复制代码
const handleClick = useCallback(() => console.log('点击'), []);
相关推荐
明月_清风1 小时前
爆破前端生态!Cloudflare 收购 Vite 背后,前端开发者会迎来什么变化?
前端·vite
星栈1 小时前
Makepad、egui、Dioxus、Tauri:Rust GUI 到底怎么选
前端·rust
ai_coder_ai1 小时前
如何在自动化脚本中实现定时操作?
java·前端·javascript
努力早日退休1 小时前
一个 9999px 引发的跨平台血案:小程序离屏隐藏元素的滚动兼容性问题
前端·javascript
YFF菲菲兔1 小时前
React 核心流程总述
react.js
嘟嘟07172 小时前
前端异步编程完全指南:从json-server到DeepSeek大模型接口调用
前端
用户059540174462 小时前
大模型多轮对话“失忆”踩坑实录:一次线上事故让我排查了48小时,最终靠 Playwright + Pytest 把记忆锁死
前端·css
橘子星2 小时前
前端薅数据神器 Fetch:不用翻墙,在线拿捏后端与 AI 接口
前端·后端
步步为营DotNet2 小时前
探索.NET 11:Blazor 在跨平台客户端应用开发的进阶实践
前端·asp.net·.net