React
在软件开发中,通常痴迷于性能提升以及如何使应用程序执行得更快,从而为用户提供更好的体验。
Memoization 是优化性能的方法之一。探讨它在 React 中的工作原理。

什么是 memoization

在解释这个概念之前,先来看一个简单的斐波那契程序:

  1. function fibonacci(n){
  2. return (n < 2) ? n : fibonacci(n-1) + fibonacci(n-2);
  3. }

显然这个算法缓慢的令人绝望,因为做了非常多的冗余计算,这个时候memoization就可以派上用场了。
简单来说,memoization 是一个过程,它允许缓存递归/昂贵的函数调用的值,以便下次使用相同的参数调用函数时,返回缓存的值而不必重新计算函数。
这确保了应用程序运行得更快,因为通过返回一个已经存储在内存中的值来避免重新执行函数需要的时间。

为什么在 React 中使用 memoization

在 React 函数组件中,当组件中的 props 发生变化时,默认情况下整个组件都会重新渲染。换句话说,如果组件中的任何值更新,整个组件将重新渲染,包括尚未更改其 values/props 的函数/组件。
看一个发生这种情况的简单示例。将构建一个基本的应用程序,告诉用户哪种酒最适合与它们选择的奶酪搭配。
将从设置两个组件开始。第一个组件将允许用户选择奶酪。然后它会显示最适合该奶酪的酒的名称。第二个组件将是第一个组件的子组件。在这个组件中,没有任何变化。将使用这个组件来跟踪 React 重新渲染的次数。
注意,本示例中使用的 classNames 来自 Tailwind CSS。
下面是父组件:<ParentComponent />

  1. // components/parent-component.js
  2. import Counts from "./counts";
  3. import Button from "./button";
  4. import { useState, useEffect } from "react";
  5. import constants from "../utils";
  6. const { MOZARELLA, CHEDDAR, PARMESAN, CABERNET, CHARDONAY, MERLOT } = constants;
  7. export default function ParentComponent() {
  8. const [cheeseType, setCheeseType] = useState("");
  9. const [wine, setWine] = useState("");
  10. const whichWineGoesBest = () => {
  11. switch (cheeseType) {
  12. case MOZARELLA:
  13. return setWine(CABERNET);
  14. case CHEDDAR:
  15. return setWine(CHARDONAY);
  16. case PARMESAN:
  17. return setWine(MERLOT);
  18. default:
  19. CHARDONAY;
  20. }
  21. };
  22. useEffect(() => {
  23. let mounted = true;
  24. if (mounted) {
  25. whichWineGoesBest();
  26. }
  27. return () => (mounted = false);
  28. }, [cheeseType]);
  29. return (
  30. <div className="flex flex-col justify-center items-center">
  31. <h3 className="text-center dark:text-gray-400 mt-10">
  32. Without React.memo() or useMemo()
  33. </h3>
  34. <h1 className="font-semibold text-2xl dark:text-white max-w-md text-center">
  35. Select a cheese and we will tell you which wine goes best!
  36. </h1>
  37. <div className="flex flex-col gap-4 mt-10">
  38. <Button text={MOZARELLA} onClick={() => setCheeseType(MOZARELLA)} />
  39. <Button text={CHEDDAR} onClick={() => setCheeseType(CHEDDAR)} />
  40. <Button text={PARMESAN} onClick={() => setCheeseType(PARMESAN)} />
  41. </div>
  42. {cheeseType && (
  43. <p className="mt-5 dark:text-green-400 font-semibold">
  44. For {cheeseType}, <span className="dark:text-yellow-500">{wine}</span>{" "}
  45. goes best.
  46. </p>
  47. )}
  48. <Counts />
  49. </div>
  50. );
  51. }

第二个组件是 <Counts /> 组件,它跟踪整个 <Parent Component /> 组件重新渲染的次数。

  1. // components/counts.js
  2. import { useRef } from "react";
  3. export default function Counts() {
  4. const renderCount = useRef(0);
  5. return (
  6. <div className="mt-3">
  7. <p className="dark:text-white">
  8. Nothing has changed here but I've now rendered:{" "}
  9. <span className="dark:text-green-300 text-grey-900">
  10. {(renderCount.current++)} time(s)
  11. </span>
  12. </p>
  13. </div>
  14. );
  15. }

下面的例子是点击奶酪名字时的效果:
2021-08-02-19-45-41-575037.gif
<ParentComponent /> 中的 <Counts /> 组件计算了因 <ParentComponent /> 的更改而强制 <Counts /> 组件重新渲染的次数。
目前,单击奶酪名字将更新显示下面的奶酪名字以及酒名。除了 <ParentComponent /> 会重新渲染,<Counts /> 组件也会重新渲染,即使其中的任何内容都没有改变。
想象一下,有一个组件显示数以千计的数据,每次用户单击一个按钮时,该组件或树中的每条数据都会在不需要更新时重新渲染。这就是 React.memo()useMemo() 提供性能优化所必需的地方。
现在,探索 React.memo 以及 useMemo()。之后将比较它们之间的差异,并了解何时应该使用一种而不是另一种。

什么是 React.memo()

React.memo() 随 React v16.6 一起发布。虽然类组件已经允许使用 PureComponentshouldComponentUpdate 来控制重新渲染,但 React 16.6 引入了对函数组件执行相同操作的能力。
React.memo() 是一个高阶组件 (HOC),它接收一个组件A作为参数并返回一个组件B,如果组件B的 props(或其中的值)没有改变,则组件 B 会阻止组件 A 重新渲染 。
将采用上面相同的示例,但在 <Counts /> 组件中使用 React.memo()。需要做的就是用 React.memo() 包裹 <Counts />组件,如下所示:

  1. import { useRef } from "react";
  2. function Counts() {
  3. const renderCount = useRef(0);
  4. return (
  5. <div className="mt-3">
  6. <p className="dark:text-white">
  7. Nothing has changed here but I've now rendered:{" "}
  8. <span className="dark:text-green-300 text-grey-900">
  9. {(renderCount.current ++)} time(s)
  10. </span>
  11. </p>
  12. </div>
  13. );
  14. }
  15. export default React.memo(Counts);

现在,当通过单击选择奶酪类型时,<Counts /> 组件将不会重新渲染。
2021-08-02-19-45-41-802996.gif
什么是 useMemo()
React.memo() 是一个 HOC,而 useMemo() 是一个 React Hook。使用 useMemo(),可以返回记忆值来避免函数的依赖项没有改变的情况下重新渲染。
为了在代码中使用 useMemo(),React 开发者有一些建议:

  • 可以依赖 useMemo() 作为性能优化,而不是语义保证
  • 函数内部引用的每个值也应该出现在依赖项数组中

对于下一个示例,将对 <ParentComponent /> 进行一些更改。下面的代码仅显示对之前创建的 <ParentComponent /> 的新更改。

  1. // components/parent-component.js
  2. .
  3. .
  4. import { useState, useEffect, useRef, useMemo } from "react";
  5. import UseMemoCounts from "./use-memo-counts";
  6. export default function ParentComponent() {
  7. .
  8. .
  9. const [times, setTimes] = useState(0);
  10. const useMemoRef = useRef(0);
  11. const incrementUseMemoRef = () => useMemoRef.current++;
  12. // uncomment the next line to test that <UseMemoCounts /> will re-render every t ime the parent re-renders.
  13. // const memoizedValue = useMemoRef.current++;
  14. // the next line ensures that <UseMemoCounts /> only renders when the times value changes
  15. const memoizedValue = useMemo(() => incrementUseMemoRef(), [times]);
  16. .
  17. .
  18. return (
  19. <div className="flex flex-col justify-center items-center border-2 rounded-md mt-5 dark:border-yellow-200 max-w-lg m-auto pb-10 bg-gray-900">
  20. .
  21. .
  22. <div className="mt-4 text-center">
  23. <button
  24. className="bg-indigo-200 py-2 px-10 rounded-md"
  25. onClick={() => setTimes(times+1)}
  26. >
  27. Force render
  28. </button>
  29. <UseMemoCounts memoizedValue={memoizedValue} />
  30. </div>
  31. </div>
  32. );
  33. }

首先,引入了非常重要的 useMemo() Hook。还引入了 useRef() Hook 来跟踪在组件中发生了多少次重新渲染。接下来,声明一个 times 状态,稍后将更新该状态来触发/强制重新渲染。
之后,声明一个 memoizedValue 变量,用于存储 useMemo() Hook 的返回值。useMemo() Hook 调用 incrementUseMemoRef 函数,它会在每次依赖项发生变化时将 useMemoRef.current 值加一,即 times 值发生变化。
然后创建一个按钮来点击更新times的值。单击此按钮将触发 useMemo() Hook,更新 memoizedValue 的值,并重新渲染 <UseMemoCounts /> 组件。
在这个例子中,还将 <Counts /> 组件重命名为 <UseMemoCounts />,它现在需要一个 memoizedValue 属性。
这是它的样子:

  1. // components/use-memo-counts.js
  2. function UseMemoCounts({memoizedValue}) {
  3. return (
  4. <div className="mt-3">
  5. <p className="dark:text-white max-w-md">
  6. I'll only re-render when you click <span className="font-bold text-indigo-400">Force render.</span>
  7. </p>
  8. <p className="dark:text-white">I've now rendered: <span className="text-green-400">{memoizedValue} time(s)</span> </p>
  9. </div>
  10. );
  11. }
  12. export default UseMemoCounts;

现在,当单击任何奶酪按钮时, memoizedValue 不会更新。但是当单击 Force render 按钮时,可以看到 memoizedValue 更新并且 <UseMemoCounts /> 组件重新渲染。
2021-08-02-19-45-42-142997.gif
如果注释掉当前的 memoizedValue行,并取消注释掉它上面的行:

  1. const memoizedValue = useMemoRef.current++;

将看到 <UseMemoCounts /> 组件在每次 <ParentComponent /> 渲染时重新渲染。

总结:React.memo()useMemo() 的主要区别

从上面的例子中,可以看到 React.memo()useMemo() 之间的主要区别:

  • React.memo() 是一个高阶组件,可以使用它来包装不想重新渲染的组件,除非其中的 props 发生变化。
  • useMemo() 是一个 React Hook,可以使用它在组件中包装函数。可以使用它来确保该函数中的值仅在其依赖项之一发生变化时才重新计算。

虽然 memoization 似乎是一个可以随处使用的巧妙小技巧,但只有在绝对需要这些性能提升时才应该使用它。Memoization 会占用运行它的机器上的内存空间,因此可能会导致意想不到的效果。