一、Hooks 组件

函数组件 的本质是函数,没有 state 的概念的,因此不存在生命周期一说,仅仅是一个 render 函数而已。

但是引入 Hooks 之后就变得不同了,它能让组件在不使用 class 的情况下拥有 state,所以就有了生命周期的概念,所谓的生命周期其实就是 useState、 useEffect() 和 useLayoutEffect() 。

即:Hooks 组件(使用了Hooks的函数组件)有生命周期,而函数组件(未使用Hooks的函数组件)是没有生命周期的

下面,是具体的 class 与 Hooks 的生命周期对应关系

为方便记忆,大致汇总成表格如下。

class 组件 Hooks 组件
constructor useState
getDerivedStateFromProps useState 里面 update 函数
shouldComponentUpdate useMemo
render 函数本身
componentDidMount useEffect
componentDidUpdate useEffect
componentWillUnmount useEffect  里面返回的函数
componentDidCatch
getDerivedStateFromError

二、单个组件的生命周期

1. 生命周期

V16.3 之前

我们可以将生命周期分为三个阶段:

分开来讲:

  1. 挂载阶段

  1. 组件更新阶段

  1. 卸载阶段

这种生命周期会存在一个问题,那就是当更新复杂组件的最上层组件时,调用栈会很长,如果在进行复杂的操作时,就可能长时间阻塞主线程,带来不好的用户体验,Fiber 就是为了解决该问题而生。

V16.3 之后

Fiber 本质上是一个虚拟的堆栈帧,新的调度器会按照优先级自由调度这些帧,从而将之前的同步渲染改成了异步渲染,在不影响体验的情况下去分段计算更新。

对于异步渲染,分为两阶段:

其中,reconciliation 阶段是可以被打断的,所以 reconcilation 阶段执行的函数就会出现多次调用的情况,显然,这是不合理的。

所以 V16.3 引入了新的 API 来解决这个问题:

  1. static getDerivedStateFromProps:该函数在挂载阶段和组件更新阶段都会执行,即每次获取新的props 或 state之后都会被执行在挂载阶段用来代替componentWillMount;在组件更新阶段配合 componentDidUpdate,可以覆盖 componentWillReceiveProps 的所有用法。

    同时它是一个静态函数,所以函数体内不能访问 this,会根据 nextProps 和 prevState 计算出预期的状态改变,返回结果会被送给 setState返回 null 则说明不需要更新 state,并且这个返回是必须的

  2. getSnapshotBeforeUpdate: 该函数会在 render 之后, DOM 更新前被调用,用于读取最新的 DOM 数据。

    返回一个值,作为 componentDidUpdate 的第三个参数;配合 componentDidUpdate, 可以覆盖componentWillUpdate的所有用法。

注意:V16.3 中只用在组件挂载或组件 props 更新过程才会调用,即如果是因为自身 setState 引发或者forceUpdate 引发,而不是由父组件引发的话,那么static getDerivedStateFromProps也不会被调用,在 V16.4 中更正为都调用。

即更新后的生命周期为:

  1. 挂载阶段

  1. 更新阶段

  1. 卸载阶段

2. 生命周期,误区

误解一:getDerivedStateFromProps 和 componentWillReceiveProps 只会在 props 改变 时才会调用

实际上,只要父级重新渲染,getDerivedStateFromProps 和 componentWillReceiveProps 都会重新调用,不管 props 有没有变化。所以,在这两个方法内直接将 props 赋值到 state 是不安全的。

// 子组件class PhoneInput extends Component {  state = { phone: this.props.phone };  handleChange = e => {    this.setState({ phone: e.target.value });  };  render() {    const { phone } = this.state;    return <input onChange={this.handleChange} value={phone} />;  }  componentWillReceiveProps(nextProps) {    // 不要这样做。    // 这会覆盖掉之前所有的组件内 state 更新!    this.setState({ phone: nextProps.phone });  }}// 父组件class App extends Component {  constructor() {    super();    this.state = {      count: 0    };  }  componentDidMount() {    // 使用了 setInterval,    // 每秒钟都会更新一下 state.count    // 这将导致 App 每秒钟重新渲染一次    this.interval = setInterval(      () =>        this.setState(prevState => ({          count: prevState.count + 1        })),      1000    );  }  componentWillUnmount() {    clearInterval(this.interval);  }  render() {    return (      <>        <p>          Start editing to see some magic happen :)        </p>        <PhoneInput phone='call me!' />         <p>          This component will re-render every second. Each time it renders, the          text you type will be reset. This illustrates a derived state          anti-pattern.        </p>      </>    );  }}
class PhoneInput extends Component {state = { phone: this.props.phone };handleChange = e => {this.setState({ phone: e.target.value });};render() {const { phone } = this.state;return <input onChange={this.handleChange} value={phone} />;}componentWillReceiveProps(nextProps) {// 不要这样做。// 这会覆盖掉之前所有的组件内 state 更新!this.setState({ phone: nextProps.phone });}
}// 父组件
class App extends Component {constructor() {super();this.state = {count: 0};}componentDidMount() {// 使用了 setInterval,// 每秒钟都会更新一下 state.count// 这将导致 App 每秒钟重新渲染一次this.interval = setInterval(() =>this.setState(prevState => ({count: prevState.count + 1})),1000);}componentWillUnmount() {clearInterval(this.interval);}render() {return (<><p>Start editing to see some magic happen :)</p><PhoneInput phone='call me!' /> <p>This component will re-render every second. Each time it renders, thetext you type will be reset. This illustrates a derived stateanti-pattern.</p></>);}
}

实例可点击这里查看

当然,我们可以在 父组件App 中 shouldComponentUpdate 比较 props 的 email 是不是修改再决定要不要重新渲染,但是如果子组件接受多个 props(较为复杂),就很难处理,而且 shouldComponentUpdate 主要是用来性能提升的,不推荐开发者操作 shouldComponetUpdate(可以使用 React.PureComponet)。

我们也可以使用 在 props 变化后修改 state

class PhoneInput extends Component {  state = {    phone: this.props.phone  };  componentWillReceiveProps(nextProps) {    // 只要 props.phone 改变,就改变 state    if (nextProps.phone !== this.props.phone) {      this.setState({        phone: nextProps.phone      });    }  }  // ...}state = {phone: this.props.phone};componentWillReceiveProps(nextProps) {// 只要 props.phone 改变,就改变 stateif (nextProps.phone !== this.props.phone) {this.setState({phone: nextProps.phone});}}// ...
}

但这种也会导致一个问题,当 props 较为复杂时,props 与 state 的关系不好控制,可能导致问题

解决方案一:完全可控的组件

function PhoneInput(props) {  return <input onChange={props.onChange} value={props.phone} />;}return <input onChange={props.onChange} value={props.phone} />;
}

完全由 props 控制,不派生 state

解决方案二:有 key 的非可控组件

class PhoneInput extends Component {  state = { phone: this.props.defaultPhone };  handleChange = event => {    this.setState({ phone: event.target.value });  };  render() {    return <input onChange={this.handleChange} value={this.state.phone} />;  }}<PhoneInput  defaultPhone={this.props.user.phone}  key={this.props.user.id}/>state = { phone: this.props.defaultPhone };handleChange = event => {this.setState({ phone: event.target.value });};render() {return <input onChange={this.handleChange} value={this.state.phone} />;}
}<PhoneInputdefaultPhone={this.props.user.phone}key={this.props.user.id}
/>

当 key 变化时, React 会创建一个新的而不是更新一个既有的组件

误解二:将 props 的值直接复制给 state

应避免将 props 的值复制给 state

constructor(props) { super(props); // 千万不要这样做 // 直接用 props,保证单一数据源 this.state = { phone: props.phone };}super(props);// 千万不要这样做// 直接用 props,保证单一数据源this.state = { phone: props.phone };
}

三、多个组件的执行顺序

1. 父子组件

  1. static getDerivedStateFromProps

  2. shouldComponentUpdate

第  阶段,此时 DOM 节点已经生成完毕,组件挂载完成,开始后续流程。先依次触发同步子组件以下函数,最后触发父组件的。

React 会按照上面的顺序依次执行这些函数,每个函数都是各个子组件的先执行,然后才是父组件的执行。

所以执行顺序是:

父组件 getDerivedStateFromProps —> 父组件 shouldComponentUpdate —> 子组件 getDerivedStateFromProps —> 子组件 shouldComponentUpdate —> 子组件 getSnapshotBeforeUpdate —>  父组件 getSnapshotBeforeUpdate —> 子组件 componentDidUpdate —> 父组件 componentDidUpdate

  1. getSnapshotBeforeUpdate()

  2. componentDidUpdate()

卸载阶段

componentWillUnmount(),顺序为 父组件的先执行,子组件按照在 JSX 中定义的顺序依次执行各自的方法

注意 :如果卸载旧组件的同时伴随有新组件的创建,新组件会先被创建并执行完 render,然后卸载不需要的旧组件,最后新组件执行挂载完成的回调。

Hooks 与 React 生命周期相关推荐

  1. React 生命周期、Hooks

    React React 生命周期 Hooks Hook 的特点 Effect Hook 自定义 hooks React 生命周期 react 生命周期是组件实例从创建运行到销毁的一系列过程 compo ...

  2. react生命周期_React生命周期事件

    react生命周期 React class components can have hooks for several lifecycle events. React类组件可以具有多个生命周期事件的挂 ...

  3. 浅谈 React 生命周期

    浅谈 React 生命周期 浅谈 React 生命周期 旧版的生命周期 新版的生命周期 详解各个生命周期函数 constructor getDerivedStateFromProps render c ...

  4. 深入react技术栈(5):React生命周期

    我是歌谣 放弃很容易 但是坚持一定很酷 微信搜一搜前端小歌谣 React生命周期 挂载和卸载过程 组件得挂载 组件得卸载 数据更新过程 整体流程 文章参考深入学习React技术栈

  5. react学习(9)----react生命周期

    react生命周期1.1.constructor() constructor()中完成了React数据的初始化,它接受两个参数 :props和context,当想在函数内部使用这两个参数时 ,需使用s ...

  6. react生命周期(自己的方式理解)

    react生命周期(自己的方式理解) react的生命周期函数有:挂载,渲染/更新,卸载 首先要知道react生命周期最初始的要有哪些?最后生命中出现的又有哪些?我本人喜欢把react比作是一个人的一 ...

  7. 你不可不知道的React生命周期

    点小蓝字加关注! 作者:kim 来源:原创 写在前面 咱今天聊的话题是React生命周期,灵感来自于最近在网上发现一篇关于react生命周期的文章,里面记录的知识点竟然与小编所get到的有出入.作为一 ...

  8. ES6中的React生命周期详解

    太长时间没写react了,有点生.重新捡起来练练手. import React,{ Component } from 'react';class Demo extends Component {con ...

  9. react生命周期方法介绍

    react生命周期 react生命周期主要包括三个阶段:初始化阶段.运行中阶段.销毁阶段 react在不同的生命周期会触发不同的钩子函数 初始化阶段 getDefaultProps() 设置组件默认的 ...

最新文章

  1. C#实现类似qq的屏幕截图程序
  2. 面试官:Java中 serialVersionUID 的作用是什么?举个例子说明
  3. Web Service 的工作原理
  4. 细数高光时刻,2020全球科技巨头如何激战AI?
  5. PageOffice,word常用接口对象--Paragraph类
  6. 根据之前发的那SQL语句查询表结构的语句做了个MSSQL实体类生成器!
  7. js笔记(一)js基础、程序结构、函数
  8. windows10升级助手_利用系统自带应用在Windows 10上实现电脑免费拨打电话
  9. JavaScript通用表单验证函数
  10. 重新解释i++和++i
  11. NumPy 1.19.3 发布,Python 科学计算包
  12. vb FindwindowEx的用法实例
  13. android音乐播放器进度条误差研究
  14. Ecshop去版权方法?如何去除Ecshop版权?
  15. WiFi 扫描结果的通知过程
  16. zabbix中文乱码的三种解决办法
  17. 存储基本概念与SAN存储
  18. 【邱锡鹏老师SMP2020教程】90页自然语言处理中的预训练模型
  19. 《电路》邱关源 思维导图 第二章 电阻电路的等效变换
  20. android连接wifi不能上网,手机已经连接wifi但无法上网的详细解决方法

热门文章

  1. 数据库连接10060_通过mysql 连接远程数据库时,输入密码后,提示10060错误
  2. CANoe-Simulation Setup简介
  3. 网赚教程之个人站长域名省钱策略
  4. 百度作业帮,美团,阿里面试经验分享,测试开发最常见的24个面试问题
  5. Android冷启动与热启动概念
  6. R/RStudio详细安装流程
  7. (附源码)计算机毕业设计ssm 保险公司售后服务管理系统
  8. 人工智能的应用计算机视觉,人工智能-计算机视觉应用
  9. 未来驾驶新标配;CarLuncher车载开发塑造智能娱乐导航系统
  10. 管理科学与工程国际期刊的排行榜(转)