以空数组作为输入的useCallback和没有第二个参数的useCallback有什么区别?

App*_*son 7 reactjs react-hooks

在尝试更好地理解React Hooks的过程中,遇到了一些意想不到的行为。我试图创建一个ref数组,并通过一个onRef函数将其推送到该数组,该函数将传递给我的<div>'s。大概每次重新渲染组件时,数组都会变得越来越大,这仅仅是因为它只是一个简单的箭头功能,而没有被记忆。

因此,然后我添加了该useCallback钩子,以确保不会多次获得相同的引用,但是令我惊讶的是,它每次重新渲染时仍调用该函数。在添加一个空数组作为第二个参数之后,引用仅按预期每个组件触发一次。

下面的代码段演示了此行为。

const Example = () => {
  const _refs = React.useRef([]);
  
  // Var to force a re-render.
  const [ forceCount, forceUpdate ] = React.useState(0);
  
  const onRef = (ref) => {
    if (ref && ref !== null) {
      console.log("Adding Ref -> Just an arrow function");
      _refs.current.push(ref);
    }
  }
  
  const onRefCallbackWithoutInputs = React.useCallback((ref) => {
    if (ref && ref !== null) {
      console.log("Adding Ref -> Callback without inputs.");
      _refs.current.push(ref);
    }
  });
  
  const onRefCallbackEmptyArray = React.useCallback((ref) => {
    if (ref && ref !== null) {
      console.log("Adding Ref -> Callback with empty array");
      _refs.current.push(ref);
    }
  }, []);
  
  React.useEffect(() => {
    console.log("Refs size: ", _refs.current.length);
  });
  
  return (
    <div>
      <div ref={onRef}/>
      <div ref={onRefCallbackWithoutInputs}/>
      <div ref={onRefCallbackEmptyArray}/>
      <div onClick={() => forceUpdate(forceCount + 1)} 
        style = {
          {
            width: '100px',
            height: '100px',
            marginTop: '12px',
            backgroundColor: 'orange'
          }
        }>
        {'Click me to update'}
       </div>
    </div>
  );
};

ReactDOM.render(<Example/>, document.getElementById('root'));
Run Code Online (Sandbox Code Playgroud)
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.8.3/umd/react.production.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.8.3/umd/react-dom.production.min.js"></script>

<div id='root' style='width: 100%; height: 100%'>
</div>
Run Code Online (Sandbox Code Playgroud)

我假设useCallback第二个参数的默认值是一个空数组。那么,什么都不给出第二个参数呢?为什么表现不同?

小智 18

useCallback具有空依赖项数组的是一个记忆函数,它不会计算更新的状态(如果其中有状态,它将使用传入的初始值useState

例如,当您希望useCallback函数使用 stateuseEffect时,您应该将其传递到其依赖项数组中,就像工作一样!这样做,每次状态发生变化时,您的整个函数都会重新安装。


小智 11

我认为对于依赖项数组,所有钩子useEffect、、、、、、背后的逻辑都是相同的,如果没有传递依赖项意味着我们传递了依赖项的空值,因此比较将始终结果为 false,并且每次都会执行内联函数useLayoutEffectuseCallbackuseMemo

如果传递空依赖项意味着没有什么可以进一步比较,因此内联函数只会执行一次。(就像我们指示 React 一样,不再进行进一步比较)。

如果数组传递了某个变量,那么它将根据变量的变化计算内联函数。

尽管内联函数的实例总是会被创建。


Ret*_*sam 10

对于useMemouseCallback(本质上只是的特例useMemo),如果第二个参数是一个空数组,则该值将被记忆一次并始终返回。

如果省略第二个参数,则该值将永远不会被记忆,并且useCallbackuseMemo不会执行任何操作。

也许在某些情况下,您可能有条件地记住:

useMemo(someValue, shouldMemoize ? [] : null)
Run Code Online (Sandbox Code Playgroud)

但在绝大多数情况下,双方的第二个参数useMemo,并useCallback应被视为强制性的。实际上,Typescript定义以这种方式对待它们

// Require a second argument, and it must be an array
function useCallback<T extends (...args: any[]) => any>(callback: T, deps: DependencyList): T;

// Second argument can be undefined, but must be explicitly passed as undefined, not omitted.
function useMemo<T>(factory: () => T, deps: DependencyList | undefined): T;
Run Code Online (Sandbox Code Playgroud)

有一个开放的pull请求,该请求增强了exhaustive-depshooks eslint规则,因此,如果省略第二个参数,它将引发皮棉错误,因此不久之后,这很可能是皮棉错误。

  • @tunesmith我不认为这与这个答案中的任何内容相矛盾:答案是说“在绝大多数情况下,你应该将数组作为第二个参数传递给`useCallback`”,它并没有说有该数组不存在为空的情况。 (2认同)