今天来讨论一下如何避免滥用 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
优化代码避免额外的计算和重复渲染,优化代码性能,过度使用反而可能会适得其反。