useCallback
Before we get to useCallback, let’s take a look at an example
// parentComponent.js import React, { useState } from "react"; import { ChildComponent } from "./childComponent"; export function ParentComponent() { console.log("ParentComponent running"); const [parentCount, setParent] = useState(1); const parentCountHandler = () => { setParent(parentCount + 1); }; const childHandler = () => { console.log("childHandler", parentCount); }; return ( <div> <div onClick={parentCountHandler}>change parentCount</div> <ChildComponent onClickChild={childHandler} /> </div> ); }Copy the code
// childComponent.js
import React from "react";
export const ChildComponent = (props) => {
console.log("ChildComponent running");
return <div onClick={props.onClickChild}>ChildComponent</div>;
};
Copy the code
When we click [change parentCount] to change the value of parentCount, the console prints the following:
ParentComponent and ChildComponent are all re render when parentCount changes. But the child ChildComponent does not use parentCount, so rerendering of the child is not what we expect. To block ChildComponent re render, we must think of PureComponent, Memo, shouldComponentUpdate, The purpose of all three methods is to control the child component’s re render by making a shallow comparison of the props passed by the parent. Next, we use the memo to process the subcomponents
import React, { memo } from "react"; Export const ChildComponent = memo((props) => {console.log("ChildComponent running"); return <div onClick={props.onClickChild}>ChildComponent</div>; });Copy the code
Unfortunately, the output is still re render for the child component. Why is that? Because the parent component is passed to the child as a function childHandler, the function is recreated each time, assigning a new memory address, so the component changes its props. OnClickChild by performing a shallow comparison. So re Render will still be executed. Now it’s time for us Callback
import React, { useCallback, useState } from "react"; import { ChildComponent } from "./childComponent"; export function ParentComponent() { console.log("ParentComponent running"); const [parentCount, setParent] = useState(1); const parentCountHandler = () => { setParent(parentCount + 1); }; Const childHandler = useCallback(() => {console.log("childHandler", parentCount); } []); return ( <div> <div onClick={parentCountHandler}>change parentCount</div> <ChildComponent onClickChild={childHandler} /> </div> ); }Copy the code
After useCallback came on, we finally got the result we wanted, with the parent component’s own re renderer 7 times and the child component standing still.
Here’s why useCallback can do it.
Because useCallback actually caches the instance of the callback function each time it is rendered, it will only be recreated if the value of the useCallback parameter changes, otherwise it will remain unchanged. The sub component does not re-render because it finds that the props value has not changed by shallow comparison. This is why useCallback and useMemo need to be used in pairs.
useReducer
Of course, in addition to using useCallback to avoid unnecessary rendering of subcomponents, we can also consider useReducer
import React, { useReducer, useState } from "react"; import { ChildComponent } from "./childComponent"; export function ParentComponent() { console.log("ParentComponent running"); const [parentCount, setParent] = useState(1); const parentCountHandler = () => { setParent(parentCount + 1); }; // useCallback const [_, childHandler] = useReducer(() => {console.log("childHandler", parentCount); }); return ( <div> <div onClick={parentCountHandler}>change parentCount</div> <ChildComponent onClickChild={childHandler} /> </div> ); }Copy the code
The above code can achieve the same effect as useCallback. Because React guarantees dispatch is always the same.
useContext
Let’s look at one more complicated example just to get us to useContext
import React, { useReducer, useState } from "react"; import { ChildComponent } from "./childComponent"; const students = []; const changeStudents = (oldData, action) => { switch (action.type) { case "add": return [...oldData, { name: action.data.name }]; case "delete": return oldData.filter((item) => item.name ! == action.data.name); default: return oldData; }}; ` export function ParentComponent() { console.log("ParentComponent running"); const [parentCount, setParent] = useState(1); const parentCountHandler = () => { setParent(parentCount + 1); }; const [studentsState, onSubmit] = useReducer(changeStudents, students); return ( <div> <div onClick={parentCountHandler}>change parentCount</div> <ChildComponent onSubmit={onSubmit} /> {studentsState.map((child) => ( <div key={child.name}>{child.name}</div> ))} </div> ); }Copy the code
import React, { memo } from "react";
import { AddComponent } from "./addComponent";
import { DeleteComponent } from "./deleteComponent";
export const ChildComponent = memo((props) => {
console.log("ChildComponent running");
return (
<div>
<AddComponent onSubmit={props.onSubmit} />
<DeleteComponent onSubmit={props.onSubmit} />
</div>
);
});
Copy the code
import React, { useRef, useState } from "react";
export const AddComponent = (props) => {
const [text, setText] = useState("");
const inputRef = useRef();
return (
<>
<input
ref={inputRef}
value={text}
onChange={(e) => setText(e.target.value)}
/>
<div
onClick={() => {
props.onSubmit({ type: "add", data: { name: text } });
inputRef.current.value = "";
}}
>
add
</div>
</>
);
};
Copy the code
import React from "react"; export const DeleteComponent = (props) => { return ( <div onClick={() => { props.onSubmit({ type: "delete", data: {name: "lei"}}); }} > delete </div> ); };Copy the code
In the code above, we implement a three-level nesting of components. The studentsState value in the ParentComponent is changed by the AddComponent and DeleteComponent callback methods passed by the top-level ParentComponent to the child component. In order to pass the callback function to the third level, I first pass the function to the second level. In real development, if there are many levels, you may have to continue to pass layers down, so you may have a big head when looking for it. Don’t worry, we haven’t introduced useContext yet, so let’s fix the following code
import React, { useReducer, useState } from "react"; import { ChildComponent } from "./childComponent"; const students = []; const changeStudents = (oldData, action) => { switch (action.type) { case "add": return [...oldData, { name: action.data.name }]; case "delete": return oldData.filter((item) => item.name ! == action.data.name); default: return oldData; }}; export const ParentContext = React.createContext(); ParentContext export function ParentComponent() {console.log("ParentComponent running"); const [parentCount, setParent] = useState(1); const parentCountHandler = () => { setParent(parentCount + 1); }; const [studentsState, onSubmit] = useReducer(changeStudents, students); Provider value={onSubmit}> // Use parentContext. Provider to pass onSubmit <div> <div OnClick ={parentCountHandler}>change parentCount</div> <ChildComponent /> // Remove the callback passed to ChildComponet {studentsState.map((child) => ( <div key={child.name}>{child.name}</div> ))} </div> </ParentContext.Provider> ); }Copy the code
import React, { memo } from "react"; import { AddComponent } from "./addComponent"; import { DeleteComponent } from "./deleteComponent"; export const ChildComponent = memo((props) => { console.log("ChildComponent running"); Return (<div> <AddComponent /> // DeleteComponent </div>); });Copy the code
import React, { useContext, useRef, useState } from "react"; import { ParentContext } from "./parentComponent"; export const AddComponent = (props) => { const [text, setText] = useState(""); const inputRef = useRef(); const onSubmit = useContext(ParentContext); OnSubmit return (<> <input ref={inputRef} value={text} onChange={(e) => setText(e.target.value)} /> <div onClick={() => { onSubmit({ type: "add", data: { name: text } }); inputRef.current.value = ""; }} > add </div> </> ); };Copy the code
import React, { useContext } from "react"; import { ParentContext } from "./parentComponent"; export const DeleteComponent = (props) => { const onSubmit = useContext(ParentContext); <div onClick={() => {onSubmit({type: "delete", data: {name: ""}}); }} > delete </div> ); };Copy the code
Verify that the execution results are the same, but it eliminates the annoyance of passing in layers of callback functions.