UsePrevious Is used to hold the state of the previous rendering.

React documentation provides an implementation:

function usePrevious(value) {
  const ref = useRef();
  useEffect(() = > {
    ref.current = value;
  });
  return ref.current;
}
Copy the code

UsePrevious records start with empty values and record status values after each render so that each render returns the value of the previous render.

React-use also uses this implementation.

Ahooks provide compare to the user, which allows the user to decide whether to update the value of usePrevious records.

import { useRef } from 'react';

export type compareFunction<T> = (prev: T | undefined, next: T) = > boolean;

function usePrevious<T> (state: T, compare? : compareFunction
       ) :T | undefined {
  const prevRef = useRef<T>();
  const curRef = useRef<T>();

  const needUpdate = typeof compare === 'function' ? compare(curRef.current, state) : true;
  if (needUpdate) {
    prevRef.current = curRef.current;
    curRef.current = state;
  }

  return prevRef.current;
}

export default usePrevious;
Copy the code

Ahooks use two refs, one for the current value and one for the previous value. But why do we do it this way? How is this implementation different from the react-use implementation? ๐Ÿค”

After some fruitless experiments, I found this issue after a random search

What? Implementation of ahooks does not comply with the usage specification ๐Ÿ˜ฏ

In the link given by the issue author, it is stated that a warning will be given when the ref value is read or modified during render, and Dan explains the reason for doing so in the reply.

Reading the value of a ref in render is the same as reading a random global variable. What value is read depends on when Render is called. If the React call had been rendered at a slightly different time, it might have gotten different results.

The ahooks implementation will be problematic in the future when React defaults to turning on Concurrent mode.

The issue author provides a demo in addition to an explanation. UsePrevious used in the demo behaves differently under StrictMode.

The demo was rendered in Legacy mode, so why did StrictMode behave differently? ๐Ÿค”

After debugging at the break point, usePrevious was called twice when entering the page, causing problems with the status of curRef and preRef records.

Search StrictMode, twice and other keywords in React Issue to find the reason, or our Dan God replied:

Issue address

Components that use StrictMode and which use Hooks are rendered twice in development mode. One of the main purposes of StrictMode is to facilitate the migration of existing projects to future React versions that use concurrent mode, which is not surprising.

So far, the battle is successful ๐Ÿ˜Š

Simple change to ahooks usePrevious implementation.

function usePrevious<T> (state: T, compare? : (prev: T |undefined, next: T) => boolean) :T | undefined {
  const ref = useRef<T>();

  useEffect(() = > {
    const needUpdate = typeof compare === 'function' ? compare(ref.current, state) : true;
    if(needUpdate) { ref.current = state; }});return ref.current;
}
Copy the code