Function is introduced
React Hooks provide enhancements to functional components, such as the ability to store state, and the ability to handle side effects. Let developers implement the same functionality without using class components. In a component, any code that doesn’t convert data to views is a side effect. For example, get dom elements, add events to DOM elements, set timers, and send Ajax requests. In class components we usually use lifecycle functions to handle side effects, in functions as components we use hooks to handle side effects.
Class component deficiencies (issues to be addressed by hooks)
- Lack of logical reuse mechanism //TODO to be added
- Class components often become too complex to maintain
Split a set of coherent business logic into multiple lifecycle functions. There are multiple unrelated business logic within a lifecycle function. 3. The class member methods do not guarantee the correctness of the this pointer
Some commonly used hook functions
1. UseState is used to introduce state to function components
In our understanding, variables defined in a function are released when the function is called, so functional components are not supposed to hold state data. With useState, functional components can save state. UseState internally uses closures to hold data. When the state changes, the component is rerendered.
The basic use is no longer described, let’s look at the details of use.
- Accepts a unique parameter, the initial value of the state. The initial value can be any data type.
- The return value is an array. The array stores the state value and the method that changes the state value. The method name convention begins with set followed by the state name.
- Methods can be called multiple times. Used to store different state values.
- The argument can be a function, the initial state is whatever the function returns, and the function will only be called once, if the initial value is a dynamic value.
function App() {
const [state, setState] = useState(() = > 100) // The argument can be a function
return (
<div className="App">
<span>{state}</span>
</div>
);
}
Copy the code
Parameter is function is used mainly when the initial value is a dynamic value.
function App(props) {
// If props is passed when calling the App component,
// If count is passed, props is used; if not, the default is used
// So we can write it like this
const propsCount = props.count || 0
const [count, setCount] = useState(propsCount)
return (
<div className="App">
<span>{count}</span>
<button onClick={()= >setCount(count + 1)}>button</button>
</div>
);
}
Copy the code
1.1 Usage details of useState
Although this would work, the code is problematic. App components are rerendered when we click the button to change state. So “const propsCount = props. Count | | 0” this code in every component to apply colours to a drawing is carried out. This doesn’t make sense because this code is getting the initial value of the state and only makes sense when the component is first rendered. This means that this code only needs to be executed once. At this point, we need to use the useState pass function.
function App(props) {
const [count, setCount] = useState(() = > {
return props.count || 0
// This code will only be executed during initial rendering
// Subsequent components will not be executed when they are re-rendered
})
return (
<div className="App">
<span>{count}</span>
<button onClick={()= >setCount(count + 1)}>button</button>
</div>
);
}
Copy the code
1.2 Set the usage details of status values
- The parameter to the set status method can be a value or a function
- The method that sets the status-value method is itself asynchronous
🌰
// The argument to the setstatus method is a function
function App(props) {
const [count, setCount] = useState(() = > props.count || 0)
function handleCount(){
setCount((count) = > {
return count + 1 // What is the return value of the function})}return (
<div className="App">
<span>{count}</span>
<button onClick={handleCount}> button </button>
</div>
);
}
Copy the code
🌰 is the method of setting the status value asynchronous or handleCount as above
function handleCount(){
setCount((count) = > {
return count + 1 // What is the return value of the function
})
document.title = count
// If setCount is synchronized, then we should get the changed count.
// If it is asynchronous, this code will not wait for setCount to finish executing,
// Then the count is the same as before
}
Copy the code
When you go back to the browser and look at the code execution, it’s clear that setCount is asynchronous. So how do we solve this? It’s very simple we just put the code in the callback function, right
function handleCount(){
setCount((count) = > {
const newCount = count + 1;
document.title = newCount
return newCount
})
}
Copy the code
2. UseReducer is another way to keep the state of function components.
The method is similar to Redux and will not be described here. Look directly at the code
import { useReducer } from "react";
export default function App () {
function reducer(state, action){
switch(action.type){
case 'increment':
return state + 1;
case 'decrement':
return state - 1
default:
return state
}
}
const [count, dispatch] = useReducer(reducer, 0);return (
<div>
<button onClick={()= >Dispatch ({type: "increment"})} > ➕</button>
<span>{count}</span>
<button onClick={()= >Dispatch ({type: 'decrement'})} > ➖</button>
</div>)}Copy the code
Advantages over useState: If one of the App’s subcomponents wants to change the count value, there is no need to pass multiple methods to modify the data, such as ➕1, ➖1, etc. We can pass the Dispatch directly to the child component. A child component can trigger either action to change state by calling the Dispatch method.
3. useContext
Simplify code for retrieving data when retrieving data across component hierarchies.
Let’s look at createContext
const appContext = createContext()
function Foo(){
return (
<appContext.Consumer>{/* Passes a callback function that takes a parameter */} {value => {return<div> {value}</div>}}</appContext.Consumer>)}function App () {
return (
<appContext.Provider value={100}>
<Foo/>
</appContext.Provider>)}export default App
Copy the code
Compare that to useContext
import { useContext } from "react";
const appContext = createContext()
function Foo(){
const value = useContext(appContext)
return (
<div>{value}</div>)}function App () {
return (
<appContext.Provider value={100}>
<Foo/>
</appContext.Provider>)}Copy the code
4. useEffect
Give functional components the ability to handle side effects, similar to life cycle functions.
4.1 Execution Time
UseEffect can be seen as a combination of componentDidMount, componentDidUpdate and componentWillUnmount.
writing | Execution time |
---|---|
useEffect(() => {}) | ComponentDidMount, componentDidUpdate |
useEffect(() => {}, []) | componentDidMount |
useEffect(() => () => {}) | ComponentDidUpdate, componentWillUnMount |
Let’s check it out:
function App () {
const [count, setCount] = useState(0);
useEffect(() = > {
console.log(123)
// You can see that after the component is mounted, the console outputs 123,
// Click the button to update the component status, and the console will output 123
})
return (
<div>
<span>{count}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
</div>)}export default App
Copy the code
function App () {
const [count, setCount] = useState(0);
useEffect(() = > {
console.log(123)
// After the component is mounted, the console outputs 123,
// Click the button to update the component status, the console does not output 123}, [])return (
<div>
<span>{count}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
</div>)}Copy the code
import ReactDOM from 'react-dom'
function App () {
const [count, setCount] = useState(0);
useEffect(() = > {
return () = > {
console.log('123')
// Click the first button to update the status console output
// Click the second button to uninstall the component console output}})return (
<div>
<span>{count}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
<button onClick={()= >ReactDOM. UnmountComponentAtNode (document. GetElementById (" root "))} > click uninstall components</button>
</div>)}Copy the code
4.2 Usage Mode
The useEffect combined with asynchronous functions is briefly introduced
.function getData(){
return new Promise((reslove) = > {
reslove({count: 0})
})
}
useEffect(async() = > {let result = await getData();
console.log(result); })...Copy the code
You can see the console will have an error, we appear to analyze the cause of the error Originally in useEffect function is a function can return, return to this function is used for cleaning operation, can be carried before unloading components, but according to the above code to write, we can turn it into an asynchronous function, The return value of the asynchronous function is a Promise object. This changes the useEffect function’s original return type, so an error is reported.
We need to useEffect a function executing itself, make the self-executing function an asynchronous function, and then use the await keyword internally
useEffect(() = >{(async() = > {await axios.get()
})()
})
Copy the code
So, modify the code as follows:
function getData(){
return new Promise((reslove) = > {
reslove({count: 0})
})
}
useEffect(() = >{(async function(){
const result = await getData();
console.log(result)
})()
})
Copy the code
5. useMemo
A useMemo behaves like a calculated attribute in a Vue. It monitors changes in a value and computes new values based on the changes. If the monitoring value does not change, it will not be recalculated even if the component is re-rendered. This behavior can help avoid expensive calculations on each render. 🌰 :
function App () {
const [count, setCount] = useState(0);
const result = useMemo(() = > {
// When I click the first button to modify bool, count does not change, and the function is not executed.
console.log(1111);
return count * 2
},[count])
const [bool, setBool] = useState(true)
return (
<div>
<div>
<span>{bool? 'true ':' false '}</span>
<button onClick={()= >setBool(! Bool)}> Click modify bool</button>
</div>
<div>
<span>{result}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
</div>
</div>)}Copy the code
6. Memo method
Performance optimization to prevent component updates if the data in the component has not changed. ShouldComponentUpdate PureComponent and shouldComponentUpdate whether the component’s data changes when the component is rerendered. If not, don’t make it hard to rerender. 🌰 Generally, clicking button changes the count value of the App component, and the Foo component is also rerendered. The following code
function Foo(){
console.log('Foo component rerendered ') // The console will have output
return (
<div>I'm a Foo component</div>)}function App () {
const [count, setCount] = useState(0);
return (
<div>
<span>{count}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
<Foo/>
</div>)}Copy the code
After introducing the Memo method, click change Count to see that the console no longer prints information. That is, the count value is changed and the Foo component is not re-rendered. This improves the performance of the application.
const Foo = memo(function Foo(){
console.log('Foo component rerendered ')
return (
<div>I'm a Foo component</div>)})function App () {
const [count, setCount] = useState(0);
return (
<div>
<span>{count}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
<Foo/>
</div>)}Copy the code
7. useCallback
Caching of functions. Make the component re-render to get the same application instance. 🌰 :
const Foo = memo(function Foo(props){
console.log('Foo component rerendered ')
return (
<div>
<div>I'm a Foo component</div>
<button onClick={props.resetCount}>reset count</button>
</div>)})function App () {
const [count, setCount] = useState(0);
const resetCount = () = > {
setCount(0)}return (
<div>
<span>{count}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
<Foo resetCount={resetCount}/>
</div>)}Copy the code
Problem: Click the button to change count, and you can see that the console is also printing information, which means the Foo component is rendering again. So why?
Because the count value is changed when the button is clicked, the App component will be re-rendered when the count value is changed. When the App component is re-rendered, a different resetCount function will be generated each time, meaning that a different resetCount function is passed to the Foo component, so the Foo component will be re-rendered. So how do we optimize this? Simply, if all App components get the same resetCount function when re-rendering, then Foo components will not be re-rendered.
const Foo = memo(function Foo(props){
console.log('Foo component rerendered ')
return (
<div>
<div>I'm a Foo component</div>
<button onClick={props.resetCount}>reset count</button>
</div>)})function App () {
const [count, setCount] = useState(0);
const resetCount = useCallback(() = > {
setCount(0)
},[setCount])
return (
<div>
<span>{count}</span>
<button onClick={()= >SetCount (count + 1)}> Click</button>
<Foo resetCount={resetCount}/>
</div>)}Copy the code
8. UserRef retrieves DOM elements & saves data (across component cycles)
8.1 Obtaining DOM Elements
The method of use is not described here
8.2 Saving Data
- Using useState to save data is the state data of the component. If you modify the state data stored in useState, the component will re-render
- Using useRef to save data, the saved data remains even if the component is re-rendered. Modifying the data saved by useRef does not trigger a re-rendering of the component
So how do you use it? 🌰
function App () {
const [count, setCount] = useState(0);
let timerId = null;
useEffect(() = > {
timerId = setInterval(() = > {
setCount(count= > count + 1)},1000)
},[])
const stopCount = () = > {
console.log(timerId)
clearInterval(timerId)
}
return (
<div>
<span>{count}</span>
<button onClick={stopCount}>Click on the</button>
</div>)}Copy the code
We click on the button to clear the timerId, but click on the button to find that the timer is not cleared and the count value is still accumulating. Let timerId = NULL, this code will be executed again, so the timerId we get is always null. Modify the code as follows:
function App () {
const [count, setCount] = useState(0);
let timerId = useRef(null);
useEffect(() = > {
timerId.current = setInterval(() = > {
setCount(count= > count + 1)},1000)
},[])
const stopCount = () = > {
console.log(timerId)
clearInterval(timerId.current)
}
return (
<div>
<span>{count}</span>
<button onClick={stopCount}>Click on the</button>
</div>)}Copy the code
This proves that using useRef to save data persists even after the component is re-rendered.