React Hooks

Stella981
• 阅读 673

React Hooks几乎在所有方面都能让我们在编程中获得好处。但是某些时候的性能问题,也需要使用一些技巧来解决。我们可以使用Hooks编写快速的应用程序,但是在动手之前需要注意一两件事。

应该使用缓存记忆吗?

在大多数情况下,React速度非常快。如果您的应用程序足够快并且没有任何性能问题,那么本文不适合您。解决"虚幻"的性能问题是一件实用的事情,在开始优化之前,请先熟悉React Profiler。

React Hooks

如果您确定了渲染速度较慢的场景,那么使用缓存记忆可能是最好的选择。

React.memo是一个性能优化工具,也是一个高级组件。它类似于React.PureComponent,但用于函数组件而不是类。如果您的函数组件在相同的Props属性下呈现相同的结果,React将会使用缓存,跳过这次渲染,并重用最后一次渲染的结果。

默认情况下,它将仅对props对象中的复杂对象进行浅层比较。如果要控制比较,还可以提供自定义比较功能作为第二个参数。

不使用缓存记忆

让我们看一个不使用缓存记忆的示例,和理解为什么这会导致性能问题。

function List({ items }) {

每次单击inc时,即使List的内容没有变化,renderApp和renderList也都会被打印输出。如果组件节点树足够大,则很容易成为性能瓶颈。我们需要减少渲染数量。

简单的缓存记忆

const List = React.memo(({ items }) => {

在此示例中,缓存记忆可以正常工作并减少渲染数量。在挂载期间,将打印输出renderApp和renderList,但单击inc时,仅输出renderApp。

记忆 & 回调函数

让我们进行一些小的修改,然后将inc按钮添加到所有列表项。需要注意的是,将回调函数传递给已记忆的组件可能会导致细微的错误。

function App() {

在这个例子中,记忆将会失败。由于我们使用的是内联函数参数,因此会为每次渲染都会创建新的引用,从而使React.memo毫无用处。在记忆组件之前,我们需要一种记忆函数本身的方法。

useCallback

幸运的是,React为此有两个内置的钩子:useMemouseCallback。useMemo用于昂贵的计算,useCallback用于传递优化的子组件所需的回调。

function App() {

在这个例子中,记忆再次失败。每次按inc时都会调用renderList。useCallback的默认行为是在传递新的函数实例时计算新值。由于内联lambda在每次渲染期间都会创建新实例,因此具有默认配置的useCallback在这里没有用。

useCallback & 输入

const inc = useCallback(() => setCount(count + 1), [count]);

useCallback接受第二个参数,即输入数组,并且仅当这些输入参数更改时,useCallback才会返回新值。在此示例中,每次count更改时,useCallback将返回新的引用。由于计数在每次渲染期间都会更改,因此useCallback将在每个渲染期间返回新值。所以此代码也不会缓存记忆。

useCallback & 输入空数组

const inc = useCallback(() => setCount(count + 1), []);

useCallback可以将一个空数组作为输入,该数组将仅调用一次内部lambda并记住该引用以供将来调用。这段代码确实说明了一点,单击任何按钮时将调用一个renderApp,主inc按钮正常工作,而内部inc按钮运行失败。

计数器将从0递增到1,此后将停止。Lambda创建一次,但是被多次调用。由于创建lambda时count为0,所以创建之后,就与以下代码完全相同:

const inc = useCallback(() => setCount(1), []);

问题的根本原因在于,我们试图同时读写状态。幸运的是,React提供了两种解决问题的方法:

具有函数更新参数的useState

const inc = useCallback(() => setCount(c => c + 1), []);

useState返回的setter可以将function用作参数,您可以在其中读取给定状态的先前值。在此示例中,缓存记忆正确运行,没有任何错误。

useReducer

const [count, dispatch] = useReducer(c => c + 1, 0);

在这种情况下,useReducer缓存记忆与useState完全一样。由于保证了dispatch在渲染之间具有相同的引用,因此不需要useCallback,这使代码更容易减少了与缓存记忆相关的错误。

useReducer vs useState

useReducer更适用于管理包含多个子组件值的状态对象,或者下一个状态取决于前一个值时。使用useReducer的常见模式是与useContext一起使用,以避免在大型组件树中显式传递回调。

我建议的经验法则是,对于只在组件内部使用的数据,主要使用useState;对于需要在父级和子级之间进行双向数据交换,则useReducer是一个更好的选择。

最后

做一个形象的概括:React.memo和useReducer是最好的朋友,React.memo和useState是有时会产生冲突并引起问题的兄弟姐妹,useCallback则是您应该始终谨慎的隔壁邻居。

本文分享自微信公众号 - 前端知否(qianduanzhifou)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
6个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Easter79 Easter79
3年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
为什么mysql不推荐使用雪花ID作为主键
作者:毛辰飞背景在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一),而是推荐连续自增的主键id,官方的推荐是auto_increment,那么为什么不建议采用uuid,使用uuid究
Python进阶者 Python进阶者
1年前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这