React 生命周期

一、旧版:

1、initialization

初始化属性和状态

2、Mounting (加载阶段:涉及3个钩子函数)

2.1 componentWillMount()

组件加载时只调用,以后组件更新不调用,整个生命周期只调用一次,此时可以修改state

2.2 render()

react最重要的步骤,创建虚拟dom,进行diff算法,更新dom树都在此进行

2.3 componentDidMount()

组件渲染之后调用,只调用一次

3、Updating(更新阶段:涉及5个钩子函数)

3.1 componentWillReceivePorps(nextProps)

组件加载时不调用,组件接受新的props时调用

3.2 shouldComponentUpdate(nextProps, nextState)

组件接收到新的props或者state时调用,return true就会更新dom(使用diff算法更新),return false能阻止更新(不调用render)

3.3 componentWillUpdata(nextProps, nextState)

组件加载时不调用,只有在组件将要更新时才调用,此时可以修改state

3.4 render()

react最重要的步骤,创建虚拟dom,进行diff算法,更新dom树都在此进行

3.5 componentDidUpdate()

组件加载时不调用,组件更新完成后调用

4、Unmounting(卸载阶段:涉及1个钩子函数)


4.1 componentWillUnmount()

组件渲染之后调用,只调用一次

基本用法:
import React from "react";
import ReactDOM from "react-dom";

/**
 * 只有类组件才有生命周期
 * 
 */
class App extends React.Component{
  static defaultProps = { //设置默认属性
    name:'leah'
  }
  constructor(props) {
    super(props)
    this.state = {count: 0}
    console.log('1.执行构造函数,设置初始状态')
  }
  //1.挂载
  componentWillMount(){
    console.log('1.1组件将要挂载到页面上')
  }
  handleClick = () =>{
    this.setState({count: this.state.count +1})
  }
  render() {
    console.log('1.2.返回虚拟dom')
    return (
      <div>
        <p>{this.state.count}</p>
        <button onClick={this.handleClick}>+</button>
        {/* <ChildCounter count={this.state.count}></ChildCounter> */
            this.state.count < 3 ? <ChildCounter count={this.state.count}></ChildCounter> : null //当count>3的时候讲子组件销毁
        }
      </div>
    )
  }
  //当虚拟dom被挂载到真实dom上之后才执行这个,这个时候才能获取到真实dom
  componentDidMount(){
    console.log('1.3.组件已经挂载到了页面上')
  }
  //2.更新
  //2.1询问组件状态是否更新
  shouldComponentUpdate(nextprops,nextState){
    console.log('2.1 询问组件状态是否更新')
    // return nextState.count % 3 === 0 //表示如果state是3的倍数就更新,否则就不更新
    return true
  }
  componentWillUpdate(){ 
    console.log('2.2组件状态将要更新')
  }
  componentDidUpdate(){
    console.log('2.2组件状态已经更新')
  }

}
class ChildCounter extends React.Component{
  componentWillReceiveProps(newprops){ //组件属性的更新多了这一步
    console.log('组件将要接收新的属性',newprops)
  }
  shouldComponentUpdate(nextprops,nextState){
    console.log('询问子组件属性是否更新')
    // return nextprops.count % 6 === 0 //表示如果state是3的倍数就更新,否则就不更新
    return true
  }
  componentWillUpdate(){
    console.log('子组件属性将要更新')
  }
  componentDidUpdate(){
    console.log('子组件属性已经更新')
  }
  componentWillUnmount(){
    console.log('子组件将要被销毁')
  }
  render(){
    console.log('childCounter render')
    return(
    <div>{this.props.count}</div>
    )
  }
}
ReactDOM.render(<App></App>,document.getElementById('root'))
/**
 * componentWillMount 先父后子 
 * componentDidMount 先子后父
 */

 二、新版生命周期

 

1、 Mounting(加载阶段:涉及4个钩子函数)

1.1 constructor()

加载的时候调用一次,可以初始化state

1.2 static getDerivedStateFromProps(props, state)

组件每次被rerender的时候,包括在组件构建之后(虚拟dom之后,实际dom挂载之前),每次获取新的props或state之后;每次接收新的props之后都会返回一个对象作为新的state,返回null则说明不需要更新state;
配合componentDidUpdate,可以覆盖componentWillReceiveProps的所有用法

1.3 render()

react最重要的步骤,创建虚拟dom,进行diff算法,更新dom树都在此进行

1.4 componentDidMount()

组件渲染之后调用,只调用一次

2、Updating(更新阶段:涉及5个钩子函数)

2.1 static getDerivedStateFromProps(props, state)

组件每次被rerender的时候,包括在组件构建之后(虚拟dom之后,实际dom挂载之前),每次获取新的props或state之后;每次接收新的props之后都会返回一个对象作为新的state,返回null则说明不需要更新state;配合componentDidUpdate,可以覆盖componentWillReceiveProps的所有用法

2.2 shouldComponentUpdate(nextProps, nextState)

组件接收到新的props或者state时调用,return true就会更新dom(使用diff算法更新),return false能阻止更新(不调用render)

2.3 render()

react最重要的步骤,创建虚拟dom,进行diff算法,更新dom树都在此进行

2.4 getSnapshotBeforeUpdate(prevProps, prevState)

触发时间: update发生的时候,在render之后,在组件dom渲染之前;返回一个值,作为componentDidUpdate的第三个参数;配合componentDidUpdate, 可以覆盖componentWillUpdate的所有用法

2.5 componentDidUpdate()

组件加载时不调用,组件更新完成后调用

3、Unmounting(卸载阶段:涉及1个钩子函数)

3.1 componentWillUnmount()

组件渲染之后调用,只调用一次
基本用法:
import React from "react";
import ReactDOM from "react-dom";

/**
 * 新的生命周期函数去掉了三个:
 * componentWillReceiveProps componentWillUpdate componentWillMount
 * 多了两个:getDerivedStateFromProps  
 * 
 */
class App extends React.Component{
  static defaultProps = { //设置默认属性
    name:'leah'
  }
  constructor(props) {
    super(props)
    this.state = {count: 0}
    console.log('1.执行构造函数,设置初始状态')
  }
  //1.挂载
  handleClick = () =>{
    this.setState({count: this.state.count +1})
  }
  render() {
    console.log('1.2.返回虚拟dom')
    return (
      <div>
        <p>{this.state.count}</p>
        <button onClick={this.handleClick}>+</button>
        <ChildCounter count={this.state.count}></ChildCounter> 
      </div>
    )
  }
  //当虚拟dom被挂载到真实dom上之后才执行这个,这个时候才能获取到真实dom
  componentDidMount(){
    console.log('1.3.组件已经挂载到了页面上')
  }
  //2.更新
  //2.1询问组件状态是否更新
  shouldComponentUpdate(nextprops,nextState){
    console.log('2.1 询问组件状态是否更新')
    // return nextState.count % 3 === 0 //表示如果state是3的倍数就更新,否则就不更新
    return true
  }
  componentDidUpdate(){
    console.log('2.2组件状态已经更新')
  }

}
class ChildCounter extends React.Component{
  state = {name: 'leah', count: 0}
  shouldComponentUpdate(nextprops,nextState){
    console.log('询问子组件属性是否更新')
    // return nextprops.count % 6 === 0 //表示如果state是3的倍数就更新,否则就不更新
    return true
  }
  componentDidUpdate(){
    console.log('子组件属性已经更新')
  }
  componentWillUnmount(){
    console.log('子组件将要被销毁')
  }
  //每当状态属性变更的时候执行此方法,可以得到最新的属性和状态
  //nextProps 代表新的属性对象 prevState 代表旧的状态对象
  static getDerivedStateFromProps(nextProps, prevState){
    if( nextProps.count % 2 === 0) {
      return {count : nextProps.count * 2}
    }else {
      return {count : nextProps.count * 3}
    }
  }
  render(){
    console.log('childCounter render')
    return(
    <div>{this.state.count}</div>
    )
  }
}

ReactDOM.render(<App></App>,document.getElementById('root'))
/**
 * componentWillMount 先父后子 
 * componentDidMount 先子后父
 */

总结:

  • React16新的生命周期弃用了componentWillMount、componentWillReceivePorps,componentWillUpdate
  • 新增了getDerivedStateFromProps、getSnapshotBeforeUpdate来代替弃用的三个钩子函数(componentWillMount、componentWillReceivePorps,componentWillUpdate)
  • React16并没有删除这三个钩子函数,但是不能和新增的钩子函数(getDerivedStateFromProps、getSnapshotBeforeUpdate)混用,React17将会删除componentWillMount、componentWillReceivePorps,componentWillUpdate
  • 新增了对错误的处理(componentDidCatch)
https://zhuanlan.zhihu.com/p/38030418 这篇文章有点内容

 

 

 

posted @ 2020-04-12 20:02  leahtao  阅读(189)  评论(0编辑  收藏  举报