react 中 memo 模块作用

`memo`是一个用于优化组件性能的高阶组件。

一、避免不必要的重新渲染

1. 浅比较机制原理

1.1 组件渲染触发条件

在 React 中,当组件的`props`或`state`发生变化时,组件会重新渲染。然而,在某些情况下,即使父组件重新渲染并传递给子组件相同的`props`,子组件也不需要重新渲染。`memo`模块通过对组件的`props`进行浅比较来解决这个问题。浅比较会检查`props`对象的引用是否发生变化,如果`props`的引用没有改变,并且组件是一个纯函数组件(没有内部状态或副作用),那么`memo`会阻止子组件的重新渲染。

1.2 代码示例

javascript 复制代码
import { memo } from "react";



const MyComponent = memo((props) => {

  return <div>{props.value}</div>;

});



export default MyComponent;

2. 性能优化效果

2.1 提升应用性能

例如:在一个列表组件中,每个列表项都是一个子组件,如果列表项的内容没有变化,使用`memo`可以防止列表项组件在父组件重新渲染时重新渲染。

2.2 减少渲染成本

每次组件重新渲染都需要重新执行组件函数,生成新的虚拟 DOM,然后与旧的虚拟 DOM 进行比较,这个过程会消耗一定的计算资源。`memo`通过阻止不必要的重新渲染,减少了这些渲染成本,尤其是在组件树较深或组件更新频繁的场景中,效果更加明显。

二、与函数组件的协同作用

1. 保持函数组件的纯洁性

1.1 纯函数组件概念

纯函数组件是指没有副作用且对于相同的输入总是返回相同输出的组件。`memo`与纯函数组件配合得很好,因为它的浅比较机制基于纯函数组件的特性。当一个函数组件被`memo`包裹后,它更符合纯函数的定义,即只要`props`不变,组件的输出(渲染结果)就不会改变。

1.2 代码示例

javascript 复制代码
import { memo } from "react";



const PureFunctionComponent = memo((props) => {

  // 这里没有副作用,只是根据传入的props渲染内容

  return <div>{props.text}</div>;

});



export default PureFunctionComponent;

2. 处理函数组件的依赖关系

2.1 函数组件中的依赖管理

例如:如果一个组件依赖于一个外部的格式化函数,当这个函数作为`props`传递给被`memo`包裹的组件时,`memo`会检测到这个`props`的变化并正确处理组件的重新渲染。

2.2 代码示例

javascript 复制代码
import { memo } from "react";



const formatText = (text) => text.toUpperCase();



const FunctionComponentWithDependency = memo((props) => {

  const formattedText = formatText(props.text);

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

});



export default FunctionComponentWithDependency;

三、与 React 其他特性的结合

1. 与 React Context 的结合使用

1.1 优化 Context 的消费组件

例如:在一个主题切换的应用中,有一个通过 Context 提供的主题颜色数据,使用`memo`包裹的组件可以在主题颜色没有直接影响其`props`时避免重新渲染。

1.2 代码示例

javascript 复制代码
import { memo, useContext, createContext } from "react";



const ThemeContext = React.createContext();



const ThemedComponent = memo((props) => {

  const theme = useContext(ThemeContext);

  // 这里只使用了与props相关的逻辑,不受ThemeContext变化的直接影响

  return <div style={{ color: props.textColor }}>{props.text}</div>;

});



export default ThemedComponent;

2. 与 React Hooks 的结合使用

2.1 配合 Hooks 实现复杂功能

例如:在一个使用`useEffect`来获取数据的组件中,`memo`可以防止组件在数据获取逻辑没有改变且`props`不变的情况下重新渲染。

2.2 代码示例

javascript 复制代码
import { memo, useState, useEffect } from "react";



const DataFetchingComponent = memo((props) => {

  const [data, setData] = useState(null);

  useEffect(() => {

    // 模拟数据获取逻辑

    const fetchData = async () => {

      const result = await fetch(props.url);

      const jsonData = await result.json();

      setData(jsonData);

    };

    fetchData();

  }, [props.url]);

  return <div>{data ? data.toString() : "Loading..."}</div>;

});



export default DataFetchingComponent;
相关推荐
AI浩19 小时前
【Labelme数据操作】LabelMe标注批量复制工具 - 完整教程
运维·服务器·前端
涔溪20 小时前
CSS 网格布局(Grid Layout)核心概念、基础语法、常用属性、实战示例和进阶技巧全面讲解
前端·css
2401_8784545320 小时前
浏览器工作原理
前端·javascript
西陵20 小时前
为什么说 AI 赋能前端开发,已经不是选择题,而是必然趋势?
前端·架构·ai编程
by__csdn21 小时前
Vue3 setup()函数终极攻略:从入门到精通
开发语言·前端·javascript·vue.js·性能优化·typescript·ecmascript
天天扭码1 天前
前端如何实现RAG?一文带你速通,使用RAG实现长期记忆
前端·node.js·ai编程
Luna-player1 天前
在前端中,<a> 标签的 href=“javascript:;“ 这个是什么意思
开发语言·前端·javascript
lionliu05191 天前
js的扩展运算符的理解
前端·javascript·vue.js
小草cys1 天前
项目7-七彩天气app任务7.4.2“关于”弹窗
开发语言·前端·javascript
奇舞精选1 天前
GELab-Zero 技术解析:当豆包联手中兴,开源界如何守住端侧 AI 的“最后防线”?
前端·aigc