React中useMemo和useCallback的作用:

一、useMemo 基本用法:

useMemo 是 React 提供的一个 Hook,用于性能优化,它通过"记忆"(memoization)计算结果来避免在每次渲染时进行不必要的复杂计算。

复制代码
const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]);

主要作用:

避免重复计算

当组件重新渲染时,useMemo 会检查依赖项数组中的值是否发生变化,只有当依赖项变化时,才会重新执行计算函数,否则直接返回之前缓存的值
2.

优化渲染性能

对于计算量大的操作,使用 useMemo 可以显著提高性能,避免在每次渲染时都执行昂贵的计算
3.

保持引用稳定

当返回对象或数组时,useMemo 可以保持引用不变(除非依赖项变化),这对于传递给子组件的 props 特别有用,可以避免不必要的子组件重新渲染

使用场景:

1. 复杂计算
复制代码
function ExpensiveComponent({ a, b }) {
  const result = useMemo(() => {
    // 复杂的计算过程
    return a * b + Math.pow(a, b) - a / b;
  }, [a, b]); // 只有当a或b变化时才重新计算

  return <div>{result}</div>;
}
2. 优化子组件渲染
复制代码
function ParentComponent({ items }) {
  const processedItems = useMemo(() => {
    return items.map(item => ({
      ...item,
      fullName: `${item.firstName} ${item.lastName}`
    }));
  }, [items]); // 只有当items变化时才重新处理

  return <ChildComponent items={processedItems} />;
}
3. 避免不必要的对象/数组创建
复制代码
function FormComponent({ onSubmit }) {
  const formConfig = useMemo(() => ({
    initialValues: { username: '', password: '' },
    validate: values => {
      const errors = {};
      if (!values.username) errors.username = 'Required';
      return errors;
    }
  }), []); // 空依赖数组表示只创建一次

  return <Form {...formConfig} onSubmit={onSubmit} />;
}

注意事项:

不要滥用

useMemo 本身也有性能开销,只对真正昂贵的计算或需要稳定引用的值使用
2.

依赖数组

确保包含所有在计算函数中使用的外部值,遗漏依赖项会导致 bug
3.

不是语义保证

React 可能在将来选择"忘记"某些记忆化的值,不要依赖 useMemo 来实现业务逻辑的正确性
4.

React.memo 的区别

useMemo 记忆的是值,React.memo 记忆的是组件

二、useCallback 基本用法:

主要作用:

useCallback 是 React 提供的用于性能优化 的 Hook,它的主要作用是缓存函数引用,避免不必要的函数重新创建。

使用场景:

1. 当函数作为依赖项时
复制代码
// 不使用 useCallback - 每次渲染都会创建新函数,导致 useEffect 重复执行
const fetchData = () => { /*...*/ };
useEffect(() => {
  fetchData();
}, [fetchData]); // 依赖项每次都会变化

// 使用 useCallback - 只有依赖变化时才创建新函数
const fetchData = useCallback(() => {
  /*...*/
}, [dependency]); // 只有当 dependency 变化时才重新创建函数
2. 当函数作为 props 传递给优化过的子组件时
复制代码
// 子组件用 React.memo 优化过
const Child = React.memo(({ onClick }) => {
  /*...*/
});

function Parent() {
  // 不使用 useCallback - 每次渲染都会导致 Child 重新渲染
  const handleClick = () => { /*...*/ };
  
  // 使用 useCallback - 避免 Child 不必要的重新渲染
  const handleClick = useCallback(() => {
    /*...*/
  }, []); // 空依赖表示函数永不变化
  
  return <Child onClick={handleClick} />;
}
3. 在自定义 Hook 中返回稳定函数时
复制代码
function useCounter() {
  const [count, setCount] = useState(0);
  
  // 返回稳定的函数引用
  const increment = useCallback(() => setCount(c => c + 1), []);
  const decrement = useCallback(() => setCount(c => c - 1), []);
  
  return { count, increment, decrement };
}

具体使用时机

  1. 函数被包含在依赖数组 (如 useEffectuseMemouseCallback 的依赖数组)中时

  2. 函数作为 prop 传递给用 React.memo 优化的子组件

  3. 函数被用作上下文值且会被多个组件使用时

  4. 函数被多次创建且创建成本较高时(虽然这种情况较少见)

不需要使用 useCallback 的情况

  1. 函数不参与任何依赖关系(没有作为 prop 传递,不在依赖数组中)
  2. 函数定义非常简单且创建成本极低
  3. 组件本身很少重新渲染

三、useMemo与 **useCallback**的关系:

useCallback(fn, deps) 相当于 useMemo(() => fn, deps),两者都用于性能优化,但:

  1. useMemo 用于记忆计算结果

  2. useCallback 专门用于记忆函数

    ​// 这两个是等价的
    const memoizedCallback = useCallback(() => doSomething(a, b), [a, b]);
    const memoizedCallback = useMemo(() => () => doSomething(a, b), [a, b]);

相关推荐
小嘟嚷ovo15 分钟前
h5,原生html,echarts关系网实现
前端·html·echarts
十一吖i1 小时前
Vue3项目使用ElDrawer后select方法不生效
前端
只可远观1 小时前
Flutter目录结构介绍、入口、Widget、Center组件、Text组件、MaterialApp组件、Scaffold组件
前端·flutter
周胡杰1 小时前
组件导航 (HMRouter)+flutter项目搭建-混合开发+分栏效果
前端·flutter·华为·harmonyos·鸿蒙·鸿蒙系统
敲代码的小吉米1 小时前
前端上传el-upload、原生input本地文件pdf格式(纯前端预览本地文件不走后端接口)
前端·javascript·pdf·状态模式
是千千千熠啊1 小时前
vue使用Fabric和pdfjs完成合同签章及批注
前端·vue.js
九月TTS2 小时前
TTS-Web-Vue系列:组件逻辑分离与模块化重构
前端·vue.js·重构
我是大头鸟2 小时前
SpringMVC 内容协商处理
前端
Humbunklung2 小时前
Visual Studio 2022 中添加“高级保存选项”及解决编码问题
前端·c++·webview·visual studio
墨水白云3 小时前
nestjs[一文学懂nestjs中对npm功能包的封装,ioredis封装示例]
前端·npm·node.js