深入理解 ReactJS:揭示重复渲染现象及其解决方案
当谈论 React 的性能问题时,不可避免的一个话题就是组件的 re-render 重复渲染。React 的组件需要关注两个阶段
- 初始阶段渲染: 当组件第一次挂载时
- 重复渲染: 组件已挂载,需要更新组件的状态
但也并不是所有的 re-render 都是有问题的,有些 re-render 是必要的,而有些 re-render 是不必要的。
必要的渲染: 当组件的状态发生变化时,需要更新。如用户和页面发生交互,或异步获取的网络新的数据等,都需要页面更新到最新的数据,这时就需要组件重复渲染。
不必要的渲染:由于工程中不合理的app架构,会导致一些组件渲染时,另外一些不需要渲染的组件,也会重复渲染,这些渲染有时是不必要的。
那在 React 中,哪些情况会引起重复渲染呢?有哪些方法可以避免一些不必要的重复渲染呢?
哪些情况导致组件重复渲染
🧐 状态变化导致重复渲染
在 React 中当组件的状态发生变化,就会重复渲染,这是 React 中组件更新的的内部机制,也是引起组件重复渲染的根本原因。

🧐 父组件导致重复渲染
当父组件重复渲染时,它的子组件都会跟着重新渲染。

🧐 Context变化导致重复渲染
当在使用 Context 时,如果 Context Provider 提供的 value 发生变化时,在所有使用 Context 数据的组件就会导致重复渲染,即使组件中只使用了 Context 中的部分数据也会导致重复渲染。

🧐 hook变化导致重复渲染
在组件中使用 hook 时,当 hook 中状态发生变化,会导致组件的重复渲染,如果在 hook 中使用了 Context 和 Context value 时,也会导致组件的重复渲染。

通过组合阻止重复渲染
⛔️不要在渲染函数中创建组件
在一个组件中的渲染函数中创建组件是最大的性能杀手,组件每一次重复渲染都会导致创建的组件销毁并重新创建,这就会比通常创建组件的性能差。

✅ 防止重复渲染 move state down
当一个组件中一部分组件使用了 state ,而另一部分组件相对和 state 相对孤立,典型的例子就是打开 关闭 dialog的组件中,通常把使用 state 的组件单独提取成一个独立的组件,这样未使用 state 的组件就不会受到 state 的变化的影响
Bad
const Component = () => {
  const [isOpen, setOpen] = useState(false)
  return (
    <div>
        <button onClick={() => setOpen(!isOpen)}>open</button>
        { isOpen && <ModalDialog />}
        {/* 状态的变化会引起 SlowComponent 重复渲染 */}
        <SlowComponent />
    </div>
  )
}
优化后
const Component = () => {
  return (
    <div>
        <ButtonWithDialog />
        <SlowComponent />
    </div>
  )
}
const ButtonWithDialog = () => {
  const [isOpen, setOpen] = useState(false)
  return (
    <>
        <button onClick={() => setOpen(!isOpen)}>open</button>
        { isOpen && <ModalDialog />}
    </>
  )
}
✅ 防止重复渲染 children as props
有时无法轻易的把一个组件单独的独立提取出来,此时可以把带状态的组件提取出来,然后把耗时的组件作为 children props 传递给那个组件,这样也可以避免重复渲染
Bad
const FullComponent = () => {
  const [state, setState] = useState(1);
  const onClick = () => {
    setState(state + 1);
  };
  return (
    <div onClick={onClick} className="click-block">
      <p>Click this component - "slow" component will re-render</p>
      <p>Re-render count: {state}</p>
      <VerySlowComponent />
    </div>
  );
};
在父组件中点击会引起父组件状态变化,父组件需要渲染,对应的 VerySlowComponent
优化后
把带状态管理的组件提取出来,接收一个 children 属性
const ComponentWithClick = ({ children }) => {
  const [state, setState] = useState(1);
  const onClick = () => {
    setState(state + 1);
  };
  return (
    <div onClick={onClick} className="click-block">
      <p>Re-render count: {state}</p>
      {children}
    </div>
  );
};x
const SplitComponent = () => {
  return (
    <>
      <ComponentWithClick>
        <>
          <p>Click the block - "slow" component will NOT re-render</p>
          <VerySlowComponent />
        </>
      </ComponentWithClick>
    </>
  );
};
✅ 防止重复渲染: components as props
和上面的情况类似,把带状态管理的组件提取出来,把相对耗时的组件作为组件的 props 传递过去,props 不受状态变化的影响,所以可以避免耗时组件的重复渲染,适用于耗时组件不受状态变化的影响,又不能作为 children 属性传递
Bad
const FullComponent = () => {
  const [state, setState] = useState(1);
  const onClick = () => {
    setState(state + 1);
  };
  return (
    <div onClick={onClick} className="click-block">
      <p>Click this component - "slow" component will re-render</p>
      <p>Re-render count: {state}</p>
      <VerySlowComponent />
      <p>Something</p>
      <AnotherSlowComponent />
    </div>
  );
};
优化后
const ComponentWithClick = ({ left, right }) => {
  const [state, setState] = useState(1);
  const onClick = () => {
    setState(state + 1);
  };
  return (
    <div onClick={onClick} className="click-block">
      <p>Re-render count: {state}</p>
      {left}
      <p>Something</p>
      {right}
    </div>
  );
};
// 把组件作为 props 传递给组件,这样耗时组件就不受点击事件的影响
const SplitComponent = () => {
  const left = (
    <>
      <h3>component with slow components passed as props</h3>
      <p>Click the block - "slow" components will NOT re-render</p>
      <VerySlowComponent />
    </>
  );
  const right = <AnotherSlowComponent />;
  return (
    <>
      <ComponentWithClick left={left} right={right} />
    </>
  );
};
使用 React.memo 避免重复渲染
使用 React.memo 可以有效的避免组件的重复渲染,但并不是使用了 React.memo 都可以避免重复渲染
✅ React.memo 中带 props 的组件
所有不是原始值的 props 都必须缓存起来,使用React.memo才能起作用,下面的例子中都使用了 React.memo ,但是第一个组件的 props 没有缓存,还是会重复渲染, 第二个由于 props 使用了缓存就不会引起重复渲染
const Child = ({ value }) => {
  console.log("Child re-renders", value.value);
  return <>{value.value}</>;
};
const ChildMemo = React.memo(Child);
const App = () => {
  const [state, setState] = useState(1);
  const onClick = () => {
    setState(state + 1);
  };
  const memoValue = useMemo(() => ({ value: "second" }), []);
  return (
    <>
      <p>first 组件还是会重复渲染</p>
      <p>Second 不会重复渲染</p>
      <button onClick={onClick}>click here</button>
      <br />
      <ChildMemo value={{ value: "first" }} />
      <br />
      <ChildMemo value={memoValue} />
    </>
  );
};
✅ React.memo中有 children 或 props作为组件时
当用 React.memo 封装的组件作为 props 或 children时,不能把 React.memo 作用到父组件上,下面的例子说明, 注意下面写法的区别
const Child = ({ value }) => {
  console.log("Child re-renders", value.value);
  return <>{value.value}</>;
};
const Parent = ({ left, children }) => {
  return (
    <div>
      {left}
      {children}
    </div>
  );
};
const ChildMemo = React.memo(Child);
const ParentMemo = React.memo(Parent);
const App = () => {
  const [state, setState] = useState(1);
  const onClick = () => {
    setState(state + 1);
  };
  
  const memoValue = useMemo(() => ({ value: "memoized" }), []);
  return (
    <>
      <button onClick={onClick}>click here</button>
      {/*虽然父组件使用 React.memo, 但是如果使用 props children 接收组件时,不起作用,点击时依然会重复渲染*/}
      <ParentMemo
        left={<Child value={{ value: "left child of ParentMemo" }} />}
      >
        <Child value={{ value: "child of ParentMemo" }} />
      </ParentMemo>
      {/* props children 传递组件,需用 React.memo 封装才能避免点击时重复渲染 */}
      <Parent left={<ChildMemo value={memoValue} />}>
        <ChildMemo value={memoValue} />
      </Parent>
    </>
  );
};
使用 useCallback useMemo
单纯的缓存 props 并不会避免子组件的重复渲染
const Child = ({ value }) => {
  console.log("Child re-renders", value.value);
  return <>{value.value}</>;
};
const App = () => {
  const [state, setState] = useState(1);
  const onClick = () => {
    setState(state + 1);
  };
  
  const memoValue = useMemo(() => ({ value: "child" }), []);
  return (
    <>
      <button onClick={onClick}>click here</button>
      <br />
      <br />
      {/* 单纯的缓存 props,Child 在点击时,依然会重复渲染 */}
      <Child value={memoValue} />
    </>
  );
};
✅ 必要的 userMemo useCallback
如果子组件使用了 React.memo 封装,那么子组件的所有的 非原始值的 props 必须缓存

如果组件在 useEffect useMemo useCallback 中使用非原始值作为依赖项 dependency ,那也应该使用缓存

避免 Context 提供的数据引起重复渲染
✅ 缓存 Provider 提供的数据

✅ 将读取,写入数据分割成不同的 Provider

✅ 将数据分割成小的 Provider

列出了常用的优化方法,如果有更好的方法,欢迎交流
相关主题
通过Vue3对比学习Reactjs: 模板语法 vs JSX Vue vs Reactjs之 props Vuejs vs Reactjs:组件之间如何通信 解密v-model:揭示Vue.js和React.js中实现双向数据绑定的不同策略 从零开始:如何在Vue.js和React.js中使用slot实现自定义内容 学习ReactJS Context: 深入理解和使用useContext
转载自:https://juejin.cn/post/7251861916146417723




