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('点击'), []);
相关推荐
Bigger1 天前
Tauri (26)——托盘图标总对不上系统主题?一行 Template Image 搞定
前端·rust·app
kyriewen1 天前
面试官问你:“AI 能写 80% 的代码了,公司为什么还需要你?”
前端·javascript·面试
甲维斯1 天前
又升级咯!坦克大战2026,科技与复古并存!
前端·人工智能·游戏开发
搬砖的码农1 天前
(08)为什么我的 Agent 一跑后台服务就卡死
前端·agent·ai编程
飘尘1 天前
前端转全栈(Java 后端)必须要知道的:开发中的锁机制与分布式并发控制
前端·后端·全栈
亲亲小宝宝鸭1 天前
前端性能监控:web-vitals
前端·性能优化·监控
陆枫Larry1 天前
可滚动页面背景填不满:`height: 100vh` vs `min-height: 100vh`
前端
Patrick_Wilson1 天前
Squash Merge 的血缘陷阱:为什么删掉的代码又活了过来
前端·git·程序员
kyriewen1 天前
今天的科技圈,全在抢英伟达的饭碗
前端·面试·ai编程