React 避坑指南:彻底搞定不必要的重新渲染

"我的页面就改了个弹窗开关,为什么整个列表都闪了一下?"

------ 如果你也曾这样怀疑人生,这篇内容就是为你写的。


一、为什么"重新渲染"这么贵?

React 的默认行为是:父组件一更新,所有子组件无条件跟着渲染

在小型 Demo 里这没什么,但在真实业务里:

  • 列表 1000 项,每项 20 个 DOM,一次渲染就是 2 万次比对;
  • 图表、表格、富文本编辑器初始化一次要几百毫秒;
  • 低端手机掉帧、风扇狂转、用户开始骂娘。

性能优化第一性原理
"不要阻止 React 更新,而是让它根本没必要更新。"


二、10 个实战招式,招招致命

招式 一句话记忆 典型场景 代码片段
① React.memo "子组件 props 没变就别来烦我" 纯展示组件 export default React.memo(Card)
② useCallback "函数引用给我稳住" 传给子组件的回调 const onClick = useCallback(() => {}, [id])
③ useMemo "对象/数组也给我稳住" 过滤、排序、映射 const list = useMemo(() => filter(raw), [raw, key])
④ 状态下放 "状态别放爷爷组件" 弹窗、开关、表单 isOpen 放到 <Modal> 父级,而非 App 根
⑤ 拆分小组件 "大组件=大锅饭" 列表项、卡片 1000 行上帝组件拆成 20 行 Item 组件
⑥ 容器/展示 "数据层与视图层离婚" 任何页面 容器取数 → 展示 memo
⑦ Context 拆值 "Context value 别每次新对象" 主题、用户 useMemo 包 value,或拆成两个 Context
⑧ 禁用 render 新建组件 "别在函数里写函数组件" 动态渲染 组件定义放顶层
⑨ 状态管理库 "props drilling 超过 3 层就上库" 跨页面共享 Zustand / Redux / Jotai
⑩ Profiler 复盘 "先量化,再优化" 任何优化前后 React DevTools → Profiler → 录制 → 对比

三、案例复盘:一个"搜索列表"从 450 ms 到 45 ms

1. 背景

  • 商品列表 500 条,支持关键字搜索、分类筛选;
  • 每次输入字符,整个页面闪一下,搜索框掉帧;
  • 测试手机:Redmi Note 9。

2. 问题定位(Profiler)

  1. 输入字符 → 根组件 setState → 整页重新渲染;
  2. 列表项组件未 memo,每条都执行 render;
  3. 过滤函数在 render 内执行,O(n) 复杂度 × 500;
  4. 回调函数每次新建,导致 memo 失效。

3. 四步优化

Step 1 状态下放

把搜索关键字 keyword 从 App 根级放到 <SearchList> 组件,其他模块不再受牵连。

Step 2 缓存计算

js 复制代码
const filtered = useMemo(
  () => goods.filter(g => g.name.includes(keyword)),
  [goods, keyword]
);

Step 3 缓存回调

js 复制代码
const onAddCart = useCallback((id) => cart.add(id), [cart]);

Step 4 列表项 memo

js 复制代码
const Item = React.memo(({ goods, onAddCart }) => (
  <div>
    <span>{goods.name}</span>
    <button onClick={() => onAddCart(goods.id)}>加购</button>
  </div>
));

4. 结果

  • 输入字符时,仅 SearchList 与列表重新渲染,Header、Sidebar、Footer 纹丝不动;
  • 单次渲染耗时从 450 ms → 45 ms;
  • 掉帧率从 18% → 0%。

四、最容易踩的 5 个"反模式"

  1. 在 render 里写箭头函数
    <button onClick={() => handleClick(id)} />

    → 每次新引用,子组件 memo 失效。

  2. 在 render 里新建对象
    <Child style={{ color: 'red' }} />

    → 同上,用 useMemo 或提前声明常量。

  3. 把函数组件写在另一个组件内部

    见上文"新建组件类型"。

  4. 滥用 key 来"刷新"组件

    只有身份变更才改 key,局部数据更新请用状态。

  5. Context value 直接传对象字面量

    js 复制代码
    <AuthCtx value={{ user, setUser }}>  

    → 每次渲染都生成新对象,所有消费者重渲染。

    解决:

    js 复制代码
    const value = useMemo(() => ({ user, setUser }), [user]);

五、checklist:上线前必查的 8 个问题

  1. 所有纯展示组件都包 React.memo 了吗?
  2. 传给 memo 组件的函数都用 useCallback 缓存了吗?
  3. 对象/数组 props 都用 useMemo 缓存了吗?
  4. 状态是否放在"最近公共祖先"?
  5. 列表 key 用的是稳定 ID 而不是索引?
  6. 过滤/排序放在 render 里了吗?挪进 useMemo
  7. 有 props drilling 超过 3 层吗?考虑 Context 或状态库。
  8. 用 Profiler 录制一次主流程,确认没出现"黄色长条"。

六、写在最后

优化重新渲染,就像给 React 应用做"断舍离":
少一点不必要的更新,多一点丝滑的体验。

把这 10 个招式练成肌肉记忆,你也能在 Code Review 时自信地留下那句:

"这段我 memo 过了,放心合并。"

祝你的下一次 npm start 不再闪屏,愿所有列表都如丝般顺滑。

相关推荐
程序员修心1 分钟前
CSS浮动与表格布局全解析
前端·html
POLITE324 分钟前
Leetcode 238.除了自身以外数组的乘积 JavaScript (Day 7)
前端·javascript·leetcode
光影少年29 分钟前
AI前端开发需要会哪些及未来发展?
前端·人工智能·前端框架
Vincent_Vang40 分钟前
多态 、抽象类、抽象类和具体类的区别、抽象方法和具体方法的区别 以及 重载和重写的相同和不同之处
java·开发语言·前端·ide
菩提小狗42 分钟前
小迪安全_第4天:基础入门-30余种加密编码进制&Web&数据库&系统&代码&参数值|小迪安全笔记|网络安全|
前端·网络·数据库·笔记·安全·web安全
闲蛋小超人笑嘻嘻44 分钟前
非父子通信: provide和inject
前端·javascript·vue.js
周亚鑫44 分钟前
vue3 js代码混淆
开发语言·javascript·ecmascript
止观止1 小时前
不止解构:深入掌握 ES6+ 对象与函数的高级语法糖
前端·javascript·es6
C_心欲无痕1 小时前
react - useTransition标记低优先级更新
前端·react.js·前端框架
捻tua馔...1 小时前
antd3的表单实现(HOC解决方案)
前端·javascript·react.js