旧版本,函数组件是没有生命周期的。新版本中通过useEffect触发函数的生命周期
React的组件生命周期分为
挂载阶段
、更新阶段
和销毁阶段
。因为React的state不具有Vue的响应式,所以并没有create阶段
constructor:组件被创建时调用,用于初始化状态和绑定方法。
static getDerivedStateFromProps:在组件被创建和重新渲染之前调用,用于根据props更新状态。
【已废弃】componentWillMount: 组件即将挂载,已经被废弃,向下兼容请使用【UNSAFE_componentWillMount】
render: 对组件视图结构进行编译【必须定义的一个钩子】
componentDidMount: 组件挂载完成
import { Component } from 'react'
export default class Home extends Component {
state = {
title: 'home'
}
clickUpdate = () => {
this.setState({
title: '主页'
})
}
UNSAFE_componentWillMount(): void {
console.log(this.state.title) // 'title'
console.log(document.querySelector('.home_box')) // null
}
render() {
console.log(this.state.title, 'render') // 'title'
console.log(document.querySelector('.home_box'), 'render') // null
return (
<>
<div className="home_box">{this.state.title}</div>
<button onClick={this.clickUpdate}>获取DOM</button>
</>
)
}
componentDidMount(): void {
console.log(this.state.title) // 'title'
console.log(document.querySelector('.home_box')) // home
}
}
import { Component } from 'react'
export default class Home extends Component {
state = {
title: 'home'
}
clickUpdate = () => {
this.setState({
title: '主页'
})
}
UNSAFE_componentWillMount(): void {
console.log(this.state.title) // 'title'
console.log(document.querySelector('.home_box')) // null
}
render() {
console.log(this.state.title, 'render') // 'title'
console.log(document.querySelector('.home_box'), 'render') // null
return (
<>
<div className="home_box">{this.state.title}</div>
<button onClick={this.clickUpdate}>获取DOM</button>
</>
)
}
componentDidMount(): void {
console.log(this.state.title) // 'title'
console.log(document.querySelector('.home_box')) // home
}
// 更新阶段
shouldComponentUpdate(
nextProps: Readonly<{}>,
nextState: Readonly<{ title: string }>,
nextContext: any
): boolean {
// nextProps: {}, nextState: 更新后的state, nextContext: {}
console.log(nextProps, nextState, nextContext)
if (this.state.title === nextState.title) {
// 如果数据没有发生变化 则不执行组件更新, 即是不执行render()钩子
return false
}
return true
}
// componentWillUpdate: 严格模式下eslint, 会报错
// UNSAFE_componentWillUpdate: 这个在高版本React也废弃了, 但是在ESLint模式下也可以使用[向下兼容]
UNSAFE_componentWillUpdate(
nextProps: Readonly<{}>,
nextState: Readonly<{}>,
nextContext: any
): void {
console.log(nextProps, nextState, nextContext, 'UNSAFE_componentWillUpdate')
}
componentDidUpdate(
prevProps: Readonly<{}>,
prevState: Readonly<{}>,
snapshot?: any
): void {
console.log(prevProps, prevState, snapshot, 'componentDidUpdate')
}
UNSAFE_componentWillReceiveProps(
nextProps: Readonly<{}>,
nextContext: any
): void {
console.log(nextProps, nextContext, 'UNSAFE_componentWillReceiveProps')
}
}
除了上述方法,还有一些其他的生命周期方法,如错误处理相关的方法(componentDidCatch、getDerivedStateFromError)和静态方法(getDerivedStateFromProps、getDerivedStateFromError)
需要注意的是,React 16.3之后,一些生命周期方法已经被废弃或改名,如componentWillMount、componentWillUpdate、componentWillReceiveProps等,建议使用新的方法来替代
React 18引入了一些新的生命周期方法,以便更好地管理组件的生命周期事件
useEffect
: 【异步执行的,不会阻塞渲染】用于在组件挂载、更新或卸载时执行副作用操作。它可以替代旧的生命周期方法componentDidMount、componentDidUpdate和componentWillUnmountimport { useState, useEffect } from 'react'
const Home = () => {
const [count, setCount] = useState(0)
const add = () => {
setCount(count + 1)
}
// 接收一个参数时:组件加载完成和组件状态更新时 执行
useEffect(function () {
console.log('zhw')
})
// 如果只希望函数挂载完后,执行一次,组件状态变化不再执行,可以传空数组
useEffect(function () {
console.log('zhw')
}, [])
// 接收参数二时,参数二是一个数组,用于指定依赖数据,用于当依赖数据变化时,执行一次回调
// 1、组件挂载完后执行一次;2、仅仅当count的状态更新,执行一次
useEffect(function () {
console.log('zhw')
}, [count])
// 组件卸载之前按
// 1、
useEffect(function () {
return function () {
// 该子函数会在组件 即将卸载 和 状态更新 的时候,自动执行
console.log('组件卸载了')
}
})
// 2、
useEffect(function () {
return function () {
// 该子函数会在组件 仅仅在 即将卸载 的时候,自动执行
console.log('组件卸载了')
}
}, [])
return (
<>
<span>{count}</span>
<button onClick={add}>+</button>
</>
)
}
export default Home
useLayoutEffect
: 在DOM更新之后同步执行[阻塞渲染],以确保在浏览器绘制之前执行副作用操作。适用于需要在浏览器执行绘制和布局之前立即执行的副作用操作。
import { useLayoutEffect, useState } from 'react'
const Home = () => {
const [count, setCount] = useState(0)
const add = () => {
setCount(count + 1)
}
useLayoutEffect(
function () {
console.log('useLayoutEffect')
},
[count]
)
return (
<>
<span>{count}</span>
<button onClick={add}>+</button>
</>
)
}
export default Home