Written in the beginning

React Hooks were used extensively on my last project, where we skipped the traditional class components and went straight to function components, which was a real challenge. In the process of project development, I also found that other partners in the project (including myself) sometimes used improperly, so I made a relatively comprehensive summary of several official hook functions.

The reason for functional components

The reason for functional components is that traditional class components do have some disadvantages:

  • Class componentthisIt’s pointing around a little bit
  • Organizing code by options can be painful when components are large because class components are inherently separate and do not adhere to the principle of cohesion
  • Components are not easy to reuse, especiallymixin, it is easy to bring the problem of unclear data source

Functional components are “stateful”

We know that in the past, functional components were called “dumb components” because they didn’t have their own state and were usually used to do some rendering of the view, i.e. UI = render(props). This is a pure input-output model with no side effects. But React Hooks make it possible for functional components to have their own state.

Functional components will be called many times during execution, and it is definitely not feasible to store state in the function body. Because functions are expendable things.

That is exactly what Hooks do: save the state of a function component outside the function. Specifically, the Hooks linked list for this function component. When functional components need this state, use Hooks to “hook” the state from outside the function body.

Functional components actually have a “life cycle”

The life cycle of a functional component can be divided into three parts:

First render –> re-render –> destroy

First-render is triggered when we first use functional components; If the props changes, the render function is called, triggering re-render.

Each rendering is independent. That’s the beauty of functional components.

How does React decide whether to call the Render function to update the UI view? It depends on whether the data has been updated. In terms of the entire component tree, data refers to the state of the entire component; In terms of a functional component, data can also be considered as a combination of props and its own state.

Render execution depends on data changes, and the state data in data is stored in a linked list.

What are the properties of linked lists? That is, each element has a next pointer to the next element, which is linked to each other. This is why hooks cannot be used for conditional/loop/nesting, because none of this guarantees that the hooks list is read in exactly the same order every time it is rendered. In particular, for state reads, the reading order is inconsistent with the order of the first rendering of the linked list records, which can directly result in some useState hooks reading the wrong state values.

UseSate, where the state is stored

usage

const [count, setCount] = useState(0);
Copy the code

The principle of

First, useState generates a state and a function that modifies the state. This state is stored outside of the functional component, and every time the render is re-rendered, the render goes outside and hooks the state back in, reads it as a constant and writes it into the render.

Rerendering is triggered by calling a function that changes the state. A change in props and a call to setState trigger re-render.

Since each render is independent, each render reads a separate state value. This state value is the constant read from the state hook.

This is called the Capture Value feature. Each render is independent and the state of each render is a constant.

In-depth nature

Let’s get down to the nitty-gritty and see how useState and Re-render relate:

  1. Functional components are rendered first, one by oneuseStateDo this in turn, generating a hooks list that records eachstateThe initial value of and corresponding tosetterfunction
  2. The linked list will hang outside the functional component and can beuseStateOr equivalentsetteraccess
  3. When at some point calledsetSetter“Will directly change the hooks list
  4. The hooks list is essentially the state table for this functional component, and its changes are equivalent to state changes that cause the functional component to rerender
  5. This functional component is re-rendered and executes touseStateBecause an hooks list was already mounted on the first execution, the values are read directly from the list

That’s why it’s called useState, not createState.

UseRef, DOM access and external state saving

UseRef have do

UseRef has two main functions:

  • To access DOM;
  • Used to save variables outside the current functional component.

Access to the DOM

Let’s see how the former works first:

const inputRef = useRef(null);

const handleClick = () = >{ inputRef.current? .focus(); }return (
  <input ref={inputRef} />
  <button onClick={handleClick}>Click on the</button>
)
Copy the code

This makes it easy to access DOM nodes.

Save mutable values

As mentioned earlier, useState can easily store state values, but because of the Capture Value feature of functional components, it is not possible to retrieve the changed state values in a convenient form.

const [num, setNum] = useState(0);

const increaseNum = () = > {
    setNum(prev= > prev + 1);
    console.log(num); // The old values are still printed because num is quantized in this frame
}
Copy the code

UseRef will create a ref object and store the ref object outside the functional component. This has the advantage of:

  1. Independent of thecapture valueExternal storage, don’t worry about getting stale variables;
  2. You can change the status synchronously.

Our tests are as follows:

const numRef = useRef(0);

const increaseNum = () = > {
    numRef.current += 1;
    console.log(numRef.current); // Get the latest value
}
Copy the code

But note ⚠️ : this does not cause the functional component to be rerendered because the reference is unchanged. This is an easy place to go wrong!

UseEffect, lifecycle and observer

Usage and Suggestions

The useEffect model is extremely simple, as follows:

useEffect(effectFn, deps);
Copy the code

UseEffect can simulate three life cycles of old times: componentDidMount, shouldComponentUpdate, componentWillUnmount, equivalent to three life cycles into one API.

ShouldComponentUpdate: shouldComponentUpdate: shouldComponentUpdate: shouldComponentUpdate: shouldComponentUpdate: shouldComponentUpdate: shouldComponentUpdate

useEffect(() = > {
  // xxx
});
Copy the code

ComponentDidMount (” componentDidMount “, “componentDidMount”, “componentDidMount”, “componentDidMount”, “componentDidMount”, “componentDidMount”, “componentDidMount”);

useEffect(() = > {
  // xxx} []);Copy the code

ComponentWillUnmount returns a cleanup function in effectFn as follows:

useEffect(() = > {
  // Perform side effects
  // ...
  return () = > {
    // Clear the above side effects
    // ...}; } []);Copy the code

In addition, one rule should always be followed: do not lie about DEPS dependencies. Otherwise it will cause a series of bugs. Of course the editor’s linter doesn’t allow us to do this either, which is crucial.

The principle of

EffectFn is the side effect function that is executed when a dependency changes, and the side effect, in this case, is not a derogatory term, it’s a neutral term.

Any interaction between the inside and outside of a function is a side effect, such as printing a log, starting a timer, sending a request, reading global variables, and so on.

Ok, now the effectFn can return a cleanUp function, which cleans up the side effect. Typical clearing functions, such as clearInterval and clearTimeout, are as follows:

useEffect(() = > {
  const timer = setTimeout(() = > console.log("over"), 1000);
  return () = > clearTimout(timer);
});
Copy the code

UseEffect is executed every time a render is completed, but effectFn is executed depending on whether the dependency changes. When useEffect is executed, the dependency of the current rendering is compared to the dependency of the previous rendering. If there is no change, effectFn is not executed. If there is a change, effectFn is executed.

If there is no dependency, react assumes that it changes every time it runs useEffect.

UseEffect has three typical characteristics:

  • Does not block rendering until after each rendering is complete, improving performance
  • At each runeffectFnBefore you do that, you have to run it the previous timeeffectFnThe legacy of thecleanUpThe function executes (if any)
  • When a component is destroyed, the last run is madeeffectFnThe legacy of thecleanUpThe function executes.

Object. Is is used to compare the dependencies in the deps array with those in the previous one.

Object.is(22.22); // true

Object.is([], []); // false
Copy the code

There is a danger that when the child elements in the DEps array are of reference type, each comparison will be false, thus executing effectFn. When object. is compares reference types, it compares whether two Pointers point to the same address in heap memory.

UseEffect is implemented by placing the internal effectFn in two places on the first rendering: a hooks list and an EffectList queue. After rendering, the collection of effectFn’s in the EffectList is executed.

So, to put it bluntly, re-render depends entirely on what’s inside the list.

details

In contrast to Async Mounted in vue, effectFn in useEffect should always return no or a cleanUp function. It is not possible to write something like this:

// Incorrect usage ❌
useEffect(async() = > {const response = await fetch("...");
  // ...
});
Copy the code

It is also easy to see that we do not need to rely on useEffect for the second Setter function returned by useState. In fact, React is Memoization of the Setter functions so that the Setter functions are exactly the same every time you render them. There is no need to put the Setter functions in the DEPS array.