问答文章1 问答文章501 问答文章1001 问答文章1501 问答文章2001 问答文章2501 问答文章3001 问答文章3501 问答文章4001 问答文章4501 问答文章5001 问答文章5501 问答文章6001 问答文章6501 问答文章7001 问答文章7501 问答文章8001 问答文章8501 问答文章9001 问答文章9501

React如何避免重渲染

发布网友 发布时间:2023-08-03 12:20

我来回答

1个回答

热心网友 时间:2024-10-22 10:56



组件的重新渲染

我们可以在 React 组件中的 props 和 state 存放任何类型的数据,通过改变 props 和 state,去控制整个组件的状态。当 props 和 state 发生变化时,React 会重新渲染整个组件,组件重新渲染的过程可简化如下图:



译者之前对diff的理解是,对于一个改变 props 的组件,diff能自动计算出组件内部DOM树的不同,然后经过对比,找出真正变化的DOM节点,对变化部分进行渲染。这个是错误的理解,diff算法只是用来计算出改变状态或 props的组件/虚拟节点,而这个组件/虚拟节点,无论多大,它都会重新渲染。


假设有一个渲染完成的组件,如下图:



接下来因为状态改变,需要重新渲染下图的绿色的节点,如下图:



一般的想法是只需要更新下面的三个绿色节点就能够完成组件的更新



然而!只要组件的 props 或 state 发生了变化就会重新渲染整个组件,因此除了上述的三个绿色节点以外,还需要重新渲染所有的*的节点



除了必要渲染的三个节点外,还渲染了其他不必要渲染的节点,这对性能是一个很大的浪费。如果对于复杂的页面,这将导致页面的整体体验效果非常差。因此要提高组件的性能,就应该想尽一切方法减少不必要的渲染。

shouldComponentUpdate

shouldComponentUpdate这个函数会在组件重新渲染之前调用,函数的返回值确定了组件是否需要重新渲染。函数默认的返回值是 true,意思就是只要组件的 props 或者 state 发生了变化,就会重新构建 virtual DOM,然后使用 diff 算法进行比较,再接着根据比较结果决定是否重新渲染整个组件。函数的返回值为 false 表示不需要重新渲染。


函数默认返回为 true.

PureRenderMixin

React 官方提供了 PureRenderMixin 插件,插件的功能就是在不必要的情况下让函数 shouldComponentUpdate 返回 false, 使用这个插件就能够减少不必要的重新渲染,得到一定程度上的性能提升,其使用方法如下:

import PureRenderMixin from 'react-addons-pure-render-mixin';
class FooComponent extends React.Component {
constructor(props) {
super(props);
this.shouldComponentUpdate = PureRenderMixin.shouldComponentUpdate.bind(this);
}
render() {
return <div className={this.props.className}>foo</div>;
}
}
我们需要在组件中重写 shouldComponentUpdate,PureRenderMixin源码中对PureRenderMixin.shouldComponentUpdate的定义是这样

shouldComponentUpdate(nextProps, nextState) {
return shallowCompare(this, nextProps, nextState);
}
重写的方法里面根据组件的目前的状态和组件接下来的状态进行浅比较,如果组件的状态发生变化则返回结果为 false,状态没有发生变化则返回结果为 true

shouldComponentUpdate(nextProps, nextState) {
return !shallowEqual(this.props, nextProps) ||
!shallowEqual(this.state, nextState);
}
在 React 的最新版本里面,提供了 React.PureComponent 的基础类,而不需要使用这个插件。


译者注:所以在一个较大的组件决定重渲染的时候,我们可以在每一个子组件中绑定新的shouldComponentUpdate方法,这样可以减少子组件重新渲染的次数。


我们自己可以重写 shouldComponentUpdate 这个函数,使得其能够对任何事物进行比较,也就是深比较(通过一层一层的递归进行比较),深比较是很耗时的,一般不推荐这么干,因为要保证比较所花的时间少于重新渲染的整个组件所花的时间,同时为了减少比较所花的时间我们应该保证 props 和 state 尽量简单,不要把不必要的属性放入 state,能够由其他属性计算出来的属性也不要放入 state 中。

Immutable.js

对于复杂的数据的比较是非常耗时的,而且可能无法比较,通过使用 Immutable.js 能够很好地解决这个问题,Immutable.js 的基本原则是对于不变的对象返回相同的引用,而对于变化的对象,返回新的引用。因此对于状态的比较只需要使用如下代码即可:

shouldComponentUpdate() {
return ref1 !== ref2;
}
同样需要我们在子组件中将shouldComponentUpdate方法重写。

Pure Component

如果一个组件只和 props 和 state 有关系,给定相同的 props 和 state 就会渲染出相同的结果,那么这个组件就叫做纯组件,换一句话说纯组件只依赖于组件的 props 和 state,下面的代码表示的就是一个纯组件。

render() {
return (
<div style={{width: this.props.width}}>
{this.state.rows}
</div>
);
}
如果某个子组件的 props 是固定的不会发生变化,我们叫做无状态组件。在这个组件里面使用 pureRenderMixin 插件,能够保证 shouldComponentUpdate 的返回一直为 false。所以,分清纯组件和无状态组件,在无状态组件中重写shouldComponentUpdate方法是最好的选择。

key

在写动态子组件的时候,如果没有给动态子项添加key prop,则会报一个警告。这个警告指的是,如果每一个子组件是一个数组或者迭代器的话,那么必须有一个唯一的key prop,那么这个key prop是做什么的呢?
我们想象一下,假如需要渲染一个有5000项的成绩排名榜单,而且每隔几秒就会更新一次排名,其中大部分排名只是位置变了,还有少部分是完全更新了,这时候key就发挥作用了,它是用来标识当前的唯一性的props。现在尝试来描述这一场景

[{
sid: '10001',
name: 'sysuzhyupeng'
}, {
sid: '10008',
name: 'zhyupeng'
}, {
sid: '120000',
name: 'yupeng'
}]
其中sid是学号,那么我们来实现成绩排名的榜单

import React from 'react';
function Rank({ list }){
return (
<ul>
{list.map((entry, index)=>(
<li key={index}>{entry.name}</li>
))}
</ul>
)
}
我们把key设成了序号,这么做的确不会报警告了,但这样是非常低效的做法,这个key是用来做virtual Dom diff的,上面的做法相当于用了一个随机键,那么不论有没有相同的项,更新都会重新渲染。


正确的做法非常简单,只需要把key的内容换成sid就可以了。


那么还有另一个问题,当key相同的时候,React会怎么渲染呢,答案是只渲染第一个相同key的项,且会报一个警告。

声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com
美国的联邦储备体系采用的中央银行制度是 什么是联储 打四合扣表面不能起印 四合扣四合扣安装工具 中国旅行社协会协会的主要任务 办公室的女同志都在议论ihush俏物悄语,ihush俏物悄语是什么啊? 老听有人说Ihush 俏物悄语,这个网站是干嘛的啊? ihush项链什么牌子-ihush项链掉色怎么办 姓李的要起名.用“米"或“豆字加偏旁有那些字 美国联邦储备委员会美国联邦储备委员会简介 React中有哪些类定义组件 生理氧化钠溶液可以用热水把它加热点吗? 尼尔的目的是什么 卢卡斯·尼尔的运动生涯 梨形微胖在裤装搭配在上有哪些注意事项? ...是什么现在,严重吗?该乙怎么治疗?传染性高吗? 怀孕初期梦见鱼是什么意思 用侯明昊的名字起qq昵称 朋友让我帮忙实名申请一个yy账号可靠吗 人活到怎样才算达到某种境界? 人活到什么境界可以完全不在意别人怎么看自己? 有一首韩国歌跟夜色的曲是一样的,大家知道是那首吗? 一个人的境界越高,活得就越简单 ...满了5张福卡,还能继续合成吗?如果不能合成,是不是只有送人了?_百度... 集齐五福了,红包封面还可以再得到吗? 一个面积有25平方米的房间合适多大功率的音响? 一个面积有25平方米的房间合适多大功率的音响 法律的最本质的属性 法的本质属性 零线和地线如何区分 高三复读一年能提多少分? QQ三国,JS要的元神,从1阶说到9阶,好的元神,适合的元神!要全部... qq三国仙术带什么4阶元神最好(最好便宜点) QQ三国的4阶元神升到张飞需要多少材料??? 男朋友脖子上有草莓,他说是玩游戏输了朋友捏的,可能么? 奥比岛跳跳糖薄荷草莓是什么 圣安地列斯里的FBI我都换成中国警察了但路上巡逻的警察还是美国警察要... 求圣安地列斯中国执法和警察车MOD 还有什么可以解除车门锁的?_百度知 ... 侠盗猎车手圣安地列斯怎么交警察女友 谁有圣安地列斯中国警察人物mod 侠盗飞车圣安地列斯这车的MOD谁有? 2023广东高考专科录取时间 风神AX3质量怎么样? 风神ax3质量问题爆发 台湾蓝绿阵营是什么意思啊 孙维民出演渴望城市被网友骂了吗 我的鼻子為什么總是流血,是不是快死了。。谢谢了,大神帮忙啊_百度知 ... 长年喝酒,最后流血死亡是什么病症 梦见家里的花结苞开花的预兆 我阿婆68岁一咳嗽就嗓子刺痛的疼,吃连花清瘟胶囊行吗?