当前快讯:React笔记-生命周期(七)
挂载阶段constructor生命周期值React组件从装载到卸载的全过程 在这个过程中React提供了多个内置函数供开发者在不同阶段执行需要的逻辑
状态组件由3个阶段组成 挂载阶段(MOUNTING) 更新阶段(UPDATING) 卸载阶段(UNMOUNT)
从纵向划分为2个阶段 Render阶段 Commit阶段
【资料图】
Render阶段 纯净且无副作用 会被React暂停 终止 重新启动
Commit阶段 可以使用DOM 有副作用
构造函数(这个并不是react生命周期)
触发时机
在组件初始化时触发一次
构造函数因为在组件初始化时触发 所以是初始化state最佳位置
static getDerivedStateFromProps触发时机
组件实例化后 重新渲染前
因此父组件更新 props变化 state更新 都会调用该函数
/** * * @param {*} nextProps 当前的props * @param {*} prevState 修改前的state * 该生命周期函数必须有返回值 * 它需要返回一个对象来更新 State */static getDerivedStateFromProps(nextProps, prevState) { if (nextProps.num) { return { ...prevState, num: nextProps.num * 2 } } return prevState; }
注意:
该生命周期函数是一个静态函数 所以函数内无法访问指向当前实例对象的this该生命周期函数被设计成静态方法的目的是为了保持该方法的纯粹 能够起到限制开发者访问 this 也就是实例的作用 这样就不能在里面调用实例方法或者 setState 以破坏该生命周期函数的功能原本该生命周期函数是被设计成 初始化、父组件更新 和 接收到 Props 才会触发,现在只要渲染就会触发,也就是 初始化 和 更新阶段 都会触发componentWillMount(此生命周期函数在React v17被正式废弃)
render(mounting阶段)触发时机
在构造函数和装载组件(将 DOM 树渲染到浏览器中)之间触发
注意
在此生命周期函数里使用 setState 同步设置组件内部状态 state 将不会触发重新渲染避免在该方法中引入任何的副作用(Effects)或订阅(Subscription)。对于这些使用场景,建议提前到构造函数中
渲染函数(这个并不是react生命周期)
componentDidMount作用: 仅用于渲染的纯函数 返回值取决于state和props(并不是真实渲染 返回的是一个JSX描述文件 真实渲染取决于所有组件渲染函数后)
注意
不能在函数中任何修改 state、props、请求数据等具有副作用的操作不能读写 DOM 信息不能和浏览器进行交互(例如 setTimeout)
触发时机
组件完全挂载后(插入DOM树中)立即调用 后面不会再次执行这个函数
使用场景
渲染DOM记载数据网络请求为什么在这个阶段进行网络请求
componentDidMount 这个阶段已经完成了 render,commit 阶段状态已经稳定,在这个阶段更新状态(state)不会造成冲突
实现思路
在 contructor 函数中 通过 Promise 来进行数据的请求 并且绑定到当前实例对象上 然后在 componentDidMount 中执行 Promise 把数据更新到 state 上
class Life extends React.component { componentDidMount () { }}
更新阶段componentWillReceiveProps此生命周期函数将在React v17正式废弃 推荐使用 static getDerivedStateFromProps()代替
触发时机
父组件的渲染函数被调用 不论props是否改变 在渲染函数中被渲染的子组件都会经历更新阶段并触发该生命周期函数state更新不会触发该函数
/* nextProps 当前的props*/componentWillReceiveProps(nextProps) { }
shouldComponentUpdate目前官方已经提供React.PureComponent 用来替代React.Component替代手写这个方法(不需要开发者自己实现shouldComponentUpdate,就可以进行简单的判断来提升性能)
触发时机
因为 state 和 props 变化而更新时 在重新渲染前该生命周期函数都会触发
使用场景
/** * * @param {*} nextProps 当前的props * @param {*} nextState 当前的state * @returns 根据逻辑判断返回 true 表示继续进行组件渲染,否则将停止组件渲染过程。默认返回 true,也就是说,只要组件触发了更新,组件就一定会更新 */ shouldComponentUpdate (nextProps, nextState) { return true }
React.PureComponent纯组件componentWillUpdate这个是用于继承 并不是生命周期(不需要开发者自己实现shouldComponentUpdate,就可以进行简单的判断来提升性能)
和React.Component相比 采用了props和state浅对比来实现shouldComponentUpdate 其他完全相同
存在问题(为什么PureComponent比较复杂的数据结构,可能会因深层的数据不一致而产生错误的否定判断)
这个方法只对state和props进行了新旧地址值比较 而没有深层对比其属性的变化 修改地址值属性不会导致页面更新(JavaScript 中的对象一般是可变的(Mutable)因为使用了引用赋值 新的对象简单的引用了原始对象 改变新的对象将影响到原始对象)
解决方法
避免修改原地址的属性 采用新对象参数更新(使用 shallowCopy(浅拷贝)或 deepCopy(深拷贝)来避免被修改) 但这样做造成了CPU和内存的浪费
render(updating阶段)此生命周期函数将在React v17正式废弃 使用componentWillReceiveProps代替
触发时机
更新生命周期中重新渲染之前的最后一个方法 此时已经拥有了更新后的属性和状态 并且可以在这个方法中随意处理这些数据
注意
此函数不会在初始化渲染时候被触发禁止在这个函数中调用setState方法 会造成死循环
getSnapshotBeforeUpdate渲染函数 与mounting阶段的render一致 用于渲染被React处理过的JSX到浏览器
触发时机
render渲染函数之前 state已更新
使用场景
获取render之前的dom状态
/** * * @param {*} prevProps 更新前的props * @param {*} prevState 更新前的state * @returns 返回值可以在componentDidUpdate中接收 */ getSnapshotBeforeUpdate (prevProps, prevState) { return null }
componentDidUpdate触发时机
组件每次重新渲染后触发 不会在初始化渲染的时候触发
注意
在此生命周期函数中使用setState 需要加if条件判断 prevProps、prevState 和 this.state 之间的数据变化 用于跳出循环
/** * * @param {*} prevProps 更新前的props * @param {*} prevState 更新前的state * @param {*} snapshot getSnapshotBeforeUpdate传过来的 */ componentDidUpdate (prevProps, prevState, snapshot) { }
react v17删除以下生命周期函数原因卸载阶段componentWillUnmountcomponentWillReceiveProps | componentWillUpdate | componentWillReceiveProps
原因
被废弃的三个函数都是在render之前,因为fiber的出现,很可能因为高优先级任务的出现而打断现有任务导致它们会被执行多次
简而言之就是React官方认为开发者会在这三个函数阶段编写副作用代码(约束开发者)
触发时机
在组件卸载之前触发
使用场景
注销事件监听器取消网络请求取消定时器解绑 DOM 事件
componentWillUnmount () { }
捕获错误static getDerivedStateFromError触发时机
该生命周期函数会在子孙组件抛出错误时执行
import React from "react";export default class GetDerivedStateFromError extends React.Component { /** * * @param {*} error 错误信息 * @returns * 它接收抛出的错误作为参数并且需要返回值用于更新 State */ static getDerivedStateFromError(error) { return } render () { return ( 1 ) }}
componentDidCatch错误边界(Error Boundary)触发时机
该生命周期函数会在子孙组件抛出错误时执行
使用方式和getDerivedStateFromError一致
一般情况下 任何组件在渲染期间发生错误 都会导致整个组件树都被卸载
错误边界是React的一个组件 这个组件可以捕获发生在任何组件树内错误 根据不同崩溃内容用页面反馈
注意
错误边界自身错误异步错误事件中错误服务器渲染错误
import React, { Component } from "react";export default class ErrorBoundary extends Component { state = { hasError: false }; render() { // 发生错误时 hasError 为 false 用页面反馈 if (this.state.hasError) { return 发生错误
; } // 没有发生错误 hasError 为 true 正常执行代码 return this.props.children; } // componentDidCatch生命周期函数会在子孙组件抛出错误时执行 componentDidCatch(error, info) { this.setState({ hasError: true }); }}
标签:
头条精选
- 当前快讯:React笔记-生命周期(七)2023-05-15
- AI DevOps | ChatGPT 与研发效能、效率2023-05-15
- 刘强东很“着急”,京东变革进行时2023-05-15
- 本周A股限售股解禁市值约366亿元,环比降逾2023-05-15
- 广西壮族自治区三江侗族自治县发布大雾黄色2023-05-15
- 广西壮族自治区灵川县发布大雾黄色预警_世2023-05-15
- 恩比德:绿军联盟最强 系列赛多数时候我们2023-05-15
- 恩比德:你可以说 近两战的失利削弱了76人2023-05-15
- 交行易贷通申请条件_交行易贷通2023-05-15
- 长沙南站在哪里坐地铁_长沙南站在哪里|天天2023-05-15
- 全球速递!硬核科技论 | 赛车同款序列式2023-05-15
- 今日观点!探野川西 新RAV4荣放长途试驾体验2023-05-15
- 速度与激情8演员表图片_速度与激情8演员-当2023-05-14
- 青岛会计继续教育考试可以考几次_青岛会计2023-05-14
- 持续降价 7.18万就能买宝来 销量明显稳了2023-05-14
- 通用汽车在美国召回99.5万辆汽车:气囊已致2023-05-14
- 环球今日报丨弯梁摩托车_弯梁125摩托车哪款2023-05-14
- 全球速读:飞龙骑脸怎么输黄旭东_飞龙骑脸2023-05-14
- 德昌县气象台发布大风蓝色预警信号【IV级/2023-05-14
- 热҈热҈热҈!成都下周直冲35℃!-世界新动态2023-05-14
- 凉皮的调料汁怎么调配_好吃的陕西凉皮调料2023-05-14
- 当前热文:月饼怎么做的方法教程简单_中秋2023-05-14
- 焦点热门:看一个男人,是不是真正的聪明,2023-05-14
- 当了妈妈后,还能有自己的疆域吗?2023-05-14
- 国轩高科:公司是大众SSP平台的唯一定点供2023-05-14
- 突然宣布:停售!2023-05-14
- 【环球新要闻】手机北斗导航app下载 手机2023-05-14
- 今日热闻!飞卢小说网官网小说 飞卢小说网官2023-05-14
- 全球信息:高质量发展调研行 | 第六届进博2023-05-14
- 俄罗斯当局:一架俄军直升机在俄南地区坠毁2023-05-14