React开发者的烦恼:To `useMemo` or Not to `useMemo`

今天来讨论一下如何避免滥用 React useMemo 这个 hooks。众所周知 React 有了函数式组件后 推出了一系列官方的 hooks,一部分是用来优化组件渲染,提升性能的。useMemo 就是其中之一,不使用 useMemo 或者过度使用 useMemo 都会带来性能问题。

什么是 useMemo

useMemo 是一个 React Hook,它在每次重新渲染的时候能够缓存计算的结果。

js 复制代码
const cachedValue = useMemo(calculateValue, dependencies)

参数

  • calculateValue:要缓存计算值的函数。它应该是一个没有任何参数的纯函数,并且可以返回任意类型。React 将会在首次渲染时调用该函数;在之后的渲染中,如果 dependencies 没有发生变化,React 将直接返回相同值。否则,将会再次调用 calculateValue 并返回最新结果,然后缓存该结果以便下次重复使用。
  • dependencies:所有在 calculateValue 函数中使用的响应式变量组成的数组。响应式变量包括 props、state 和所有你直接在组件中定义的变量和函数。如果你在代码检查工具中 配置了 React,它将会确保每一个响应式数据都被正确地定义为依赖项。依赖项数组的长度必须是固定的并且必须写成 [dep1, dep2, dep3] 这种形式。React 使用 Object.is 将每个依赖项与其之前的值进行比较。

返回值

在初次渲染时,useMemo 返回不带参数调用 calculateValue 的结果。

在接下来的渲染中,如果依赖项没有发生改变,它将返回上次缓存的值;否则将再次调用 calculateValue,并返回最新结果。

作用

  • 跳过代价昂贵的重新计算
  • 跳过组件的重新渲染
  • 记忆另一个 Hook 的依赖
  • 记忆一个函数

好了现在我们已经了解了什么是 useMemo,接下来看一下看一下具体使用情况。

什么情况下使用 useMemo

下面几种情况适用于使用 useMemo

复杂计算

在上面 useMemo 的一个作用就是跳过昂贵的重新计算。如下代码所示:

js 复制代码
function complexCalculation(list) {
  // 复杂计算
  return result;
}

function MyComponent({ list }) {
  const memoizedResult = useMemo(() => {
    return complexCalculation(list)
  }, [list]);

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

保持引用等价

上面提到作用之一是避免组件重新渲染。useMemo 有助于在将对象或数组作为道具传递时保持相同的引用,从而避免不必要的渲染。如下所示:

js 复制代码
function MyComponent({ values }) {
  const memoizedValues = useMemo(() => values, [values]);

  return <MemoizedChildComponent values={memoizedValues} />;
};

function ChildComponent({ values }) {
  // ...
});

const MemoizedChildComponent = React.memo(ChildComponent)

对大列表进行过滤或排序

当处理大型数据集时,useMemo 可以用于过滤或排序可以防止这些操作在每次渲染期间重新运行,从而节省宝贵的计算能力。如下所示:

js 复制代码
function MyComponent({ list }) {
  const filteredList = useMemo(() => {
    return list.filter(item => item.active);
  }, [list]);

  return <ListDisplay items={filteredList} />;
};

什么情况下不使用 useMemo

有时候过度使用 useMemo 反而会适得其反。

简单计算

对于基本计算,引入useMemo可能会使事情变得复杂而不是有帮助。如下所示:

js 复制代码
function MyComponent({ value }) {
  // 不推荐使用简单计算
  const multipliedValue = useMemo(() => {
    return value * 2;
  }, [value]);

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

记录原始值

记录原始值(例如数字或字符串)通常是多余的,因为它们不会导致不必要的渲染。如下所示:

js 复制代码
function MyComponent({ value }) {
  // 不推荐用于原始值
  const memoizedValue = useMemo(() => value + 1, [value]);

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

// 依赖关系频繁变化

// 如果输入变化很大,`useMemo`就会失去效率,因为它最终会经常重新计算。

function MyComponent({ frequentlyChangingValue }) {
  // 不建议频繁更改依赖项
  const memoizedValue = useMemo(() => {
    return expensiveCalculation(frequentlyChangingValue);
  }, [frequentlyChangingValue]);

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

小结

hooks 再好也不能贪多哦,要适当使用 useMemo 优化代码避免额外的计算和重复渲染,优化代码性能,过度使用反而可能会适得其反。

相关推荐
木叶丸5 分钟前
跨平台方案该如何选择?
android·前端·ios
LL.。8 分钟前
同步回调和异步回调
开发语言·前端·javascript
网络点点滴12 分钟前
Vue如何处理数据、v-HTML的使用及总结
前端·vue.js·html
运维咖啡吧13 分钟前
周一才上线的网站,单单今天已经超过1000访问了
前端·程序员·ai编程
世界哪有真情16 分钟前
用虚拟IP扩容端口池:解决高并发WebSocket端口耗尽问题
前端·后端·websocket
前端世界17 分钟前
打造一个可维护、可复用的前端权限控制方案(含完整Demo)
前端
LeQi23 分钟前
当!important成为代码毒瘤:你的项目是不是也中了招?
前端·css·程序员
玲小珑24 分钟前
Next.js 教程系列(九)增量静态再生 (ISR):动态更新的静态内容
前端·next.js
Mintopia33 分钟前
B 样条曲线:计算机图形学里的 “曲线魔术师”
前端·javascript·计算机图形学
前端小巷子35 分钟前
跨域问题解决方案:CORS(跨域资源共享)
前端·网络协议·面试