React性能提升

2023-10-29

了解react如何提升性能将有助于我们更好的编写代码。个人认为react中很多的性能优化,其实都是围绕着react的核心diff算法来展开的,通过优化,减少diff算法中一些不必要的步骤,从而来提高性能。下面是我平时开发总结出来的一些经验。

Component与PureComponent

一般来讲,很多人都是使用的Component,但是会带来一个问题,那就是在父组件的props或者state发生改变的时候,或者说的再准确一点,每当我们调用setState或者props发生改变的时候,都会再执行一次render,会重新生成virtualdom,这个时候,比如说在该父组件中还存在HomeArea两个组件,也会重新进行渲染(执行render -> 生成virtualDom -> 对比新旧virtualDom生成差异对象 -> 将生成的差异对象应用到真实dom上),但是如果HomeArea没有任何改变那么差异对象不包含任何东西,那么就相当于重新渲染的时候前三步耗费了性能。

这种性能耗费其实是没必要的, 于是react提供了两种方法给我们来优化这个过程。

Component中shouldComponentUpdate方法

该方法在Component中没有继承实现,需要我们手动实现。并且在初次渲染和调用forceUpdate的时候不触发

  • shouldComponentUpdate(nextProps, nextState)

如果我们要判断该组件是否更新必须通过比较this.propsnextProps加上this.statenextState, 判断他们是否改变,因为默认是返回true的。如果改变了可以通过返回true来告诉react,需要进行前面说到的重新渲染的过程,否者返回false。那么shouldComponentUpdate后面的生命周期render() -> getSnapshotBeforeUpdate() -> componentDidUpdate()都不会执行。这样就达到了提高性能的作用。

但是需要注意的一点是,虽然该组件没有被re-rendering,但是如果该组件中还包含子组件,如果此时子组件的state(这里我没有说还有props)发生了改变,此子组件还是会被重新渲染,并不是说父组件的shouldComponentUpdate返回false,所有的子组件就不会重新渲染了。你可能现在会有点迷糊,不是说返回false就不会重新渲染也不会执行render的吗? 那怎么会重新渲染子组件呢? 让我们来想一想:如果通过比较发现该组件的stateprops都没有发生改变,说明传给子组件的props肯定没有变,此时唯一可能改变的就是子组件自生的state。在开发中可能会出现这种情况,即父组件状态没有改变,但是子组件自身的state发生改变。 而且在实际项目开发中,结合使用redux来管理数据的时候,如果父组件的状态没有发生改变,子组件其实可能还是会受到props的影响的。

还有一点是在比较的过程中,如果过于复杂也会导致过多的性能损耗,还不如不进行比较,进行默认的步骤。例如深度遍历、JSON.stringify这种耗费大量的性能的操作。通常比较推荐的做法是进行浅比较,并且下面要介绍的PureComponent也是使用的浅比较,什么是浅比较在下面PureComponent专题中说。

使用PureComponent

PureComponent内部帮我们实现了shouldComponentUpdate,其他和Component一样。但是在shouldComponentUpdate进行的是一个浅比较,看看官方文档是怎么说的。

clipboard.png
他这里并没有仔细说浅比交到底是怎么比较。经过我的调试,我发现浅比较具有以下特点。

  1. 对于state只比较第一层,并且如果是基本类型,那么比较他们的值,如果是引用值,那么比较他们的引用。
  2. 对于传递来的props, 也只比较第一层,规则和上面一样。

下面给几个例子:假设state中的数据结构是这样

  constructor (props) {
    super(props)
    this.state = {
      count: 0,
      person: {
        // name: 'longjincen'
        person: {
          person: {
            name: 'longjincen'
          }
        }
      }
    }
  }

当我们继承的是Component的时候,只要调用setState就会触发重新渲染,但是当我们继承的是PureComponent的时候,就不一样了。当我们像下面这样改变state的时候

  handleClick = () => {
    const { person } = this.state
    person.person.person = {
      name: 'xiaoya'
    }
    this.setState({
      person: person
    })
  }

这个时候最外层的person始终都是同一个引用,所以这个时候是不会触发重新渲染的,就算我们内部的person是一个新的引用,因为它只比较第一层。所以当我们想要更新的时候,第一层返回的必须是一个新的对象,才会触发重新渲染。所以官方文档中也说了,如果数据结构比较复杂,那么可能会导致一些问题,要么当你知道改变的时候调用forceUpdate,要么使用immutable来包装你的state,immutable是一个js库,用法和js差不多,是facebook开发出来的,一般和PureComponent搭配使用。

所以当我们使用PureComponent的时候返回的state必须是一个新的对象,下面给出几种解决办法:

  1. 使用Object.assgin({}, xxx), 来创建一个新的对象返回
  2. 使用扩展运算符{ ...xxx }
1.
  handleClick = () => {
    const { person } = this.state
    person.person.person.name = 'xiaoya'
    this.setState({
      person: Object.assign({}, person)
    })
  }
2.
  handleClick = () => {
    const { person } = this.state
    person.person.person.name = 'xiaoya'
    this.setState({
      person: { ...person }
    })
  }

而对于接收到的前后的props的比较和上面介绍的state比较一样,但是需要注意的一种情况是,如果我们将上面的第三层的person对象传给子组件,那么在子组件中,由于接收到的person对象前后引用发生了改变,所以尽管父组件不会触发重新渲染,子组件还是会触发,而如果我们传递的是第二层或者第一层的person由于引用没有改变,子组件也不会触发重新渲染。

最后要说的是在实际项目开发中往往是使用PureComponentimmutable搭配的方式

其他

无状态组件

无状态组件就是使用定义函数的方式来定义组件,这种组件相比于使用类的方式来定义的组件(有状态组件),少了很多初始化过程,更加精简,所以要是可以使用无状态组件应当尽可能的使用无状态组件,会大幅度提升效率

不要在render中生成新的引用

定义函数、使用内联样式或者动态生成一些不依赖statepropsjsx这些,凡是不依赖stateprops的都应该提到render之外,否则会造成每次render的时候生成新的引用,会导致在diff算法对比属性或节点的过程中发现两个引用不一致,对于react来说,这说明属性值发生了改变,最后会被替换成新的引用,造成性能浪费。例如函数应该放在render函数外定义,样式应该单独建一个文件,然后引入,使用生命周期函数或者自定义函数动态生成一些不依赖statepropsjsx

始终由用户触发改变state

render函数中,只能通过特定的用户事件来触发state改变,否者容易造成不断刷新的死循环,像下面这样就是错误的

render () {
    this.setState({
        name: 'xcacsa'
    })
}

始终使用key值

避免使用数组下标作为key值, 应该确保唯一。在diff算法中,key值用来保证当列表中节点顺序改变的时候节点的复用,而不是全部替换。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

React性能提升 的相关文章

随机推荐

  • 运维必学

    欢迎关注 全栈工程师修炼指南 设为 星标 每天带你 基础入门 到 进阶实践 再到 放弃学习 专注 企业运维实践 网络安全 系统运维 应用开发 物联网实战 全栈文章 等知识分享 花开堪折直须折 莫待无花空折枝 作者 lt 安全开发运维 gt
  • VS2012编译安装VTK-5.10.1(支持Python)

    1 源码下载 到参考资料 1 下载vtk 5 10 1 zip和vtkdata 5 10 1 zip 2 源码解压 这里以D 盘为例进行说明 在D 盘中创建一个目录VTK 然后在其中创建4个目录 source build data和inst
  • mysql explain详解

    转自 http www blogjava net persister archive 2008 10 27 236813 html 在 explain的帮助下 您就知道什么时候该给表添加索引 以使用索引来查找记录从而让select 运行更快
  • Nginx反向代理与负载均衡

    文章目录 一 网关 代理与反向代理的关系 二 反向代理在系统架构中的应用场景 三 Nginx反向代理配置 1 不重定向配置 2 重定向配置 四 基于反向代理的负载均衡器 不支持https 五 负载均衡介绍 1 负载均衡策略 2 负载均衡调度
  • 三个java超级变态逻辑循环编程题

    1 有一根27厘米的细木杆 在第3厘米 7厘米 11厘米 17厘米 23厘米这五个位置上各有一只蚂蚁 木杆很细 不能同时通过一只蚂蚁 开始时 蚂蚁的头朝左还是朝右是任意的 它们只会朝前走或调头 但不会后退 当任意两只蚂蚁碰头时 两只蚂蚁会同
  • 【4】测试用例设计-判定表法

    判定表适用于有几个原因 导致几个结果的情况 实际测试中 如果输入条件较多 再加上各种输入与输出之间相互的作用关系 画出的因果图会比较复杂 容易使人混乱 为了避免这种情况 人们往往使用决策表法代替因果图法 决策表也称为 判定表 其实质就是一种
  • 各大公司薪资

    联合利华 MKT 9500 3000元安家费 普通职位 8KX12 联合利华销售代表 底薪加提成 总体一般 一般能拿到5K以上 宝洁 本8600 硕9700 博10500发14个月 11年数据 欧莱雅 MKT 6 6K X 13 11年数据
  • ajax工作原理 网页从输入url到呈现过程(TCP ,渲染引擎) 头像上传 下拉加载 节流 防抖 常见状态码

    Ajax工作原理 1 http网络传输协议 规定 前后端交互的 数据传输格式 协议 规定 前后端交互的数据传输格式 2 http协议组成两个部分 2 1前端 必须发送 请求报文格式 2 2后端 必须响应 响应报文格式 3 请求报文格式组成
  • VueX是什么?好处?何时使用?

    VueX相关 1 VueX是什么 2 使用VueX统一管理状态的好处 3 什么样的数据适合存储到Vuex中 1 VueX是什么 VueX是实现组件全局状态 数据 管理的一种机制 可以方便的实现组件之间数据的共享 如果没有VueX实现数据间的
  • 点云配准(四) Sparse Point Registration 算法浅析

    Sparse Point Registration SPR 是一篇2017年的点云配准算法 该算法的主要目的是对稀疏点云进行配准 并且取得了不错的成果和突破 本文一方面是对SPR配准算法模型进行了简单的原理解析以及附加代码实现 另一方面是对
  • c++ 的 multiple definition of `XXX‘

    文章目录 一 为什么会有多重定义问题 二 一些错误情景 1 头文件忘记加条件编译 2 类的静态变量在头文件定义 3 全局变量在 h文件定义 4 待补充 三 参考链接 一 为什么会有多重定义问题 声明 指出存储类型 并给出存储单元指定名称 定
  • C语言博客作业--嵌套循环

    一 PTA实验作业 题目1 7 1 查询水果价格 给定四种水果 分别是苹果 apple 梨 pear 桔子 orange 葡萄 grape 单价分别对应为3 00元 公斤 2 50元 公斤 4 10元 公斤 10 20元 公斤 首先在屏幕上
  • JavaScript函数的命名方式

    函数的命名方式 JavaScript代码服用单位是函数 函数可以包含一段可执行代码 也可以接受调用者传入的参数 JavaScript定义函数主要有以下三种方式 第一种方式 命名函数 第二种 匿名函数
  • 【C++】函数的设计与使用(三)重载函数、模板函数

    重载函数 参数列表不相同 可能是参数类型不相同 或者参数个数不相同 都不相同 也可以 的两个或多个函数 可以拥有相同的函数名称 编译器会把实参和每个重载函数的形参比对 找出哪个重载函数合适 所以每个重载函数的参数列表必须和其他的重载函数的不
  • ubuntu安装bochs,nasm

    1 ubuntu上安装bochs nasm 1 1 安装缘由 最近想自己做个操作系统玩一玩巩固巩固知识 工欲善其事 必先利其器 开发操作系统首先得搭建环境 编程语言上我选择C和汇编完成 开发环境是在我装的一个虚拟机ubuntu上 ubunt
  • 【满分】【华为OD机试真题2023 JAVA&JS】统计匹配的二元组个数

    华为OD机试真题 2023年度机试题库全覆盖 刷题指南点这里 统计匹配的二元组个数 知识点数组 时间限制 1s 空间限制 32MB 限定语言 不限 题目描述 给定两个数组A和B 若数组A的某个元素A i 与数组B中的某个元素B j 满足 A
  • NodeJS - Express使用

    文章目录 1 参数 1 1 获取URL中的动态参数 2 静态资源 2 1 挂载路径前缀 3 nodemon 4 1路由 4 1 路由的匹配过程 4 2 模块化路由 4 3 为路由模块添加前缀 5 中间件 5 1 全局生效的中间件 5 2 全
  • 功能强大,但因安全隐患被企业禁用的Python内置函数

    功能强大 但却因安全隐患被企业禁用的Python内置函数 eval 函数是Python的内置函数 功能非常强大 但是存在不小的安全隐患 有些企业或项目出于安全考虑 禁止使用eval 函数 会在一些安全相关的扫描校验中进行识别和拦截 杜绝使用
  • 【嵌入式】虚拟机未能将管道连接到虚拟机: 系统找不到指定的文件

    这两天虚拟机莫名奇妙的爆出这个错误 在升级win11过后解决嘞这个问题 但是win11确实不好用最后退回win10这个问题又出现了 这里记录一下我的解决办法 设置为管理员运行程序 然后遇到新的报错了 进入控制面板选择C 2015修复环境 到
  • React性能提升

    了解react如何提升性能将有助于我们更好的编写代码 个人认为react中很多的性能优化 其实都是围绕着react的核心diff算法来展开的 通过优化 减少diff算法中一些不必要的步骤 从而来提高性能 下面是我平时开发总结出来的一些经验