性能和渲染(Render)正相关

React 基于虚拟 DOM 和高效 Diff 算法的完美配合,实现了对 DOM 最小粒度的更新。大多数情况下,React 对 DOM 的渲染效率足以我们的业务日常。但在个别复杂业务场景下,性能问题依然会困扰我们。此时需要采取一些措施来提升运行性能,其很重要的一个方向,就是避免不必要的渲染(Render)

渲染(Render)时影响性能的点

React 处理 render 的基本思维模式是每次一有变动就会去重新渲染整个应用。在 Virtual DOM 没有出现之前,最简单的方法就是直接调用 innerHTML。Virtual DOM 厉害的地方并不是说它比直接操作 DOM 快,而是说不管数据怎么变,都会尽量以最小的代价去更新 DOM。React 将 render 函数返回的虚拟 DOM 树与老的进行比较,从而确定 DOM 要不要更新、怎么更新。当 DOM 树很大时,遍历两棵树进行各种比对还是相当耗性能的,特别是在顶层 setState 一个微小的修改,默认会去遍历整棵树。尽管 React 使用高度优化的 Diff 算法 ,但是这个过程仍然会损耗性能。

我自己是一名从事了多年开发的web前端老程序员,目前辞职在做自己的web前端私人定制课程,今年年初我花了一个月整理了一份最适合2019年学习的web前端学习干货,各种框架都有整理,送给每一位前端小伙伴,想要获取的可以关注我的头条号并在后台私信我:前端,即可免费获取。

渲染(Render)何时会被触发

○ 组件挂载

React 组件构建并将 DOM 元素插入页面的过程称为挂载。当组件首次渲染的时候会调用 render,这个过程不可避免。

○ setState() 方法被调用

setState 是 React 中最常用的命令,通常情况下,执行 setState 会触发 render。但是这里有个点值得关注,执行 setState 的时候一定会重新渲染吗?答案是不一定。当 setState 传入 null 的时候,并不会触发 render ,可以运行下面的 Demo 来佐证:

class App extends React.Component { state = { a: 1 }; render() { console.log("render"); return ( 

{this.state.a}

{ this.setState({ a: 1 }); // 这里并没有改变 a 的值 }} > Click me this.setState(null)}>setState null ); }}

父组件重新渲染

只要父组件重新渲染了,即使传入子组件的 props 未发生变化,那么子组件也会重新渲染,进而触发 render。

我们对上面的 demo 进行稍微的修改,可以看出当点击按钮的时候,Child 组件的 props并没有发生变化,但是也触发了 render 方法:

const Child = () => { console.log("child render"); return 
child

;};class App extends React.Component { state = { a: 1 }; render() { console.log("render"); return (

{this.state.a}

{ this.setState({ a: 1 }); }} > Click me this.setState(null)}>setState null ); }}

优化 Render 我们能做什么?

上文描述的 React 组件渲染机制其实是一种较好的做法,很好地避免了在每一次状态更新之后,需要去手动执行重新渲染的相关操作。鱼和熊掌不可兼得,带来方便的同时也会存在一些问题,当子组件过多或者组件的层级嵌套过深时,因为反反复复重新渲染状态没有改变的组件,可能会增加渲染时间又会影响用户体验,此时就需要对 React 的 render 进行优化。

上面说了不必要的 render 会带来性能问题,因此我们的主要优化思路就是减少不必要的 render。

○ shouldComponentUpdate 和 PureComponent

在 React 类组件中,可以利用 shouldComponentUpdate 或者 PureComponent 来减少因父组件更新而触发子组件的 render,从而达到目的。shouldComponentUpdate 来决定是否组件是否重新渲染,如果不希望组件重新渲染,返回 false 即可。

在 React 中 PureComponet 的源码为

if (this._compositeType === CompositeTypes.PureClass) { shouldUpdate = !shallowEqual(prevProps, nextProps) || ! shallowEqual(inst.state, nextState);}

看函数名就能够理解,PureComponet 通过对 props 和 state 的浅比较结果来实现 shouldComponentUpdate,当对象包含复杂的数据结构时,可能就不灵了,对象深层的数据已改变却没有触发 render。

看到这里,顺便看一下 shallowEqual 是如何实现的。

const hasOwnProperty = Object.prototype.hasOwnProperty;/** * is 方法来判断两个值是否是相等的值,为何这么写可以移步 MDN 的文档 * https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Object/is */function is(x: mixed, y: mixed): boolean { if (x === y) { return x !== 0 || y !== 0 || 1 / x === 1 / y; } else { return x !== x && y !== y; }}function shallowEqual(objA: mixed, objB: mixed): boolean { // 首先对基本类型进行比较 if (is(objA, objB)) { return true; } if (typeof objA !== 'object' || objA === null || typeof objB !== 'object' || objB === null) { return false; } const keysA = Object.keys(objA); const keysB = Object.keys(objB); // 长度不相等直接返回false if (keysA.length !== keysB.length) { return false; } // key相等的情况下,再去循环比较 for (let i = 0; i < keysA.length; i++) { if ( !hasOwnProperty.call(objB, keysA[i]) || !is(objA[keysA[i]], objB[keysA[i]]) ) { return false; } } return true;}

○ 利用高阶组件

在函数组件中,并没有 shouldComponentUpdate 这个生命周期,可以利用高阶组件,封装一个类似 PureComponet 的功能

const shouldComponentUpdate = arePropsEqual => BaseComponent => { class ShouldComponentUpdate extends React.Component { shouldComponentUpdate(nextProps) { return arePropsEqual(this.props, nextProps) } render() { return  } } ShouldComponentUpdate.displayName = `Pure(${BaseComponent.displayName})`; return ShouldComponentUpdate;}const Pure = BaseComponent => { const hoc = shouldComponentUpdate( (props, nextProps) => !shallowEqual(props, nextProps) ) return hoc(BaseComponent);}

使用 Pure 高阶组件的时候,只需要对我们的子组件进行装饰即可。

import React from 'react';const Child = (props) => 
{props.name}

;export default Pure(Child);

○ 使用 React.memo

React.memo 是 React 16.6 新的一个 API,用来缓存组件的渲染,避免不必要的更新,其实也是一个高阶组件,与 PureComponent 十分类似,但不同的是, React.memo 只能用于函数组件 。

基本用法

import { memo } from 'react';function Button(props) { // Component code}export default memo(Button);

高级用法

默认情况下其只会对 props 做浅层对比,遇到层级比较深的复杂对象时,表示力不从心了。对于特定的业务场景,可能需要类似 shouldComponentUpdate 这样的 API,这时通过 memo 的第二个参数来实现:

function arePropsEqual(prevProps, nextProps) { // your code return prevProps === nextProps;}export default memo(Button, arePropsEqual);

注意:与 shouldComponentUpdate 不同的是,arePropsEqual 返回 true 时,不会触发 render,如果返回 false,则会。而 shouldComponentUpdate 刚好与其相反。

○ 合理拆分组件

微服务的核心思想是:以更轻、更小的粒度来纵向拆分应用,各个小应用能够独立选择技术、发展、部署。我们在开发组件的过程中也能用到类似的思想。试想当一个整个页面只有一个组件时,无论哪处改动都会触发整个页面的重新渲染。在对组件进行拆分之后,render 的粒度更加精细,性能也能得到一定的提升。

总结

本文主要介绍了如何减少不必要的 render 来提升 React 的性能。在实际开发过程中,前端性能问题可能并不常见,随着业务的复杂度增加,遇到性能问题的概率也会随之增加。

  • 减少 render 的次数 类组件可以使用 shouldComponentUpdate 或 PureComponent,函数组件可以利用高级组件的特性或者 React.memo
  • 对组件进行合理的拆分

在摸索这些解决方案的同时,我们能够学习到诸多经典的编程思想,从而更加合理的运用框架、技术解决业务问题。

原文链接:https://www.zoo.team/article/react-render

作者:政采云前端团队

react页面数据过多怎么办_性能!!让你的 React 组件跑得再快一点,收藏相关推荐

  1. react页面数据过多怎么办_解决 React-Native 加载数据页面卡顿问题

    一般情况下,我们在componentDidMount方法里面执行请求数据.根据字面意思:页面组件加载完毕,然后请求数据. 对的 ,没毛病!!! 然而问题就出现,当我们跳转到新页面时,到界面请求完数据, ...

  2. 不挂载 组件渲染_让你的 React 组件性能跑得再快一点「实践」

    作者:天泽 转发链接:https://www.zoo.team/article/react-render 性能和渲染(Render)正相关 React 基于虚拟 DOM 和高效 Diff 算法的完美配 ...

  3. react页面数据过多怎么办_React-多页面应用

    1初始化项目 npm init create-react-app my-app 2.修改index import React from 'react'; import ReactDOM from're ...

  4. react月份选择控件_看我的案例:用react写一个日历控件!

    说明 个人娱乐所写,UI方面参照其他人的UI设计 支持功能 初始化日期 高亮'今天'以及选择日期 历史记录选择日期 支持tag标识 支持选择日期回调 支持上一月/下一月选择回调 支持tags动态切换 ...

  5. Element - table固定列页面数据过多滚动时显示问题

    背景:使用el-tabe设置右侧固定列 fixed="right",没有设置表格固定高,数据过多时滚动table会出现如下情况: 代码如下: <el-table :data= ...

  6. react中如何注释代码_学习在您的React / JavaScript代码中发现红旗?

    react中如何注释代码 by Donavon West 由Donavon West 学习在您的React / JavaScript代码中发现红旗? (Learn to spot red flags ...

  7. react 拖拽连接插件_一款精美的 react 后台管理系统

    众里寻他千百度,蓦然回首,那人却在,灯火阑珊处.这么好的react开源项目,值得珍藏了! 项目依赖模块 项目是用create-react-app创建的,依赖包如下: react(是一个用于构建用户界面 ...

  8. 共享单车数据爬取_走出低谷的共享单车,别再被数据造假的青桔给毁了!

    10月26日,有媒体报道称,在刚过去的10月23日,青桔单车全天总订单量达到2300万,超过美团和哈啰当天订单总量. 恰恰是在同日,又有网上流传的聊天截图显示,某城市管理群中青桔运维自爆主管让其刷单的 ...

  9. drawforeground只有鼠标事件进入才刷新_为什么移动鼠标会让操作系统跑得更快?...

    作者丨InfoQ 译者丨无明 策划丨小智 1 有人在 Stack Overflow 上问了一个问题 在玩 Hypnospace Outlaw(催眠帝国的法外狂徒)这款游戏时,我发现移动鼠标会让网页加载 ...

最新文章

  1. leetcode10 为什么p[j-1] == '*'的时候,不能用递推公式dp[i][j] = dp[i][j-1] || dp[i][j-2] || dp[i-1][j]
  2. jQuery一些常用特效方法使用实例
  3. J2EE项目工具集(转)
  4. Python之路【第八篇】:Python模块
  5. 大师之路-GoLive 视频教程
  6. IT人回家过年的尴尬
  7. 江陵中学2021高考成绩查询,2021高考序幕拉开 荆州38080名考生赴考
  8. vc++中画线时xor_C ++'xor_eq'关键字和示例
  9. 一键就绪的VMware Cloud Foundation
  10. 【零基础学Java】—throw关键字(四十六)
  11. Ubuntu18 安装yum
  12. 【转】kubernetes 中 deployment 支持哪些键值
  13. Team viewer 未就绪,请检查您的连接
  14. 兄弟连php视频网址
  15. 含有一个量词的命题的否命题_高一 | 数学必修一全称量词与存在量词知识点总结...
  16. html好看的后台页面布局,25 个精美的后台管理界面模板和布局
  17. 联通物联卡为什么没有网络_联通物联卡的资费和优势怎样
  18. bongo cat mver手机版|键盘猫手机版
  19. MOOC中国接受《环球时报》英文版专访:MOOC证书对求职有用吗?
  20. linux 查看java进城_linux查看java进程|线程池信息

热门文章

  1. ecshop商品详细描述调用商品相册代码
  2. arcgis for android 学习 - (4) 了解mapView的一些方法和事件
  3. 在线旅游资源点评受宠,但质量参差不齐
  4. python print函数用法_Python3.2中Print函数用法实例详解
  5. java dateformat 线程安全_SimpleDateFormat线程安全问题深入解析
  6. php接口调用教程,php接口调用
  7. sqlmap 连接mysql_sqlmap从入门到精通-第三章-4-4-使用sqlmap直连MSSQL获取webshell或权限...
  8. C语言playsoundw函数,使用inline hook实现修改PC微信通知铃声-哥哥微信来了
  9. 这些动物,你认识几个呢
  10. 2021年人工神经网络第三次作业-第二题:遗传算法与人工神经网络-参考答案