当前位置: 首页 > news >正文

React hooks(一):useState

1.React hooks

React hooks是React16.8的新特性,可以让React函数组件具有状态,并提供类似componentDidMount和componentDidUpdate等生命周期方法。

React 早期版本,类组件可以在shouldComponentUpdate中,通过判断props和state是否发生变化来决定需不需要重新渲染组件。而继承PureComponent后,当组件更新时,如果组件的 props 和 state 都没发生改变, render 方法就不会触发,省去 Virtual DOM 的生成和比对过程,达到提升性能的目的。具体就是 React 自动在shouldComponentUpdate中帮我们做了一层浅比较。这样我们可以少写 shouldComponentUpdate 函数。相对于函数组件,我们知道函数每次调用,就会执行函数体,然而React 官网没有提供对应的方法来缓存函数组件以减少一些不必要的渲染,直到 16.6 出来的 React.memo函数。

React 16.8 新出来的Hooks可以让React 函数组件具有状态,并提供类似 componentDidMount和componentDidUpdate等生命周期方法。

Hook 这个单词的意思是"钩子"。

React Hooks 的意思是,组件尽量写成纯函数,如果需要外部功能和副作用,就用钩子把外部代码"钩"进来。 React Hooks 就是那些钩子。

你需要什么功能,就使用什么钩子。React 默认提供了一些常用钩子,你也可以封装自己的钩子。

所有的钩子都是为函数引入外部功能,所以 React 约定,钩子一律使用use前缀命名,便于识别。你要使用 xxx 功能,钩子就命名为 usexxx。

2.useState

鉴于纯函数自身无法保存状态,要在两次组件函数执行期间保存和修改状态,必须引入副作用,使用外部存储能力,所以引入了useState。

const [state, setState] = useState(initialState);

其中,初始化参数initialState可以试一个值,也可以是一个函数。如果传递函数作为 initialState,则它将被视为 初始化函数。它应该是纯函数,不应该接受任何参数,并且应该返回一个任何类型的值。当初始化组件时,React 将调用你的初始化函数,并将其返回值存储为初始状态。

set函数的参数可以是一个值nextState,也可以是一个函数。如果你将函数作为 nextState 传递,它将被视为 更新函数。它必须是纯函数,只接受待定的 state 作为其唯一参数,并应返回下一个状态。React 将把你的更新函数放入队列中并重新渲染组件。在下一次渲染期间,React 将通过把队列中所有更新函数应用于先前的状态来计算下一个状态。

2.1 必须组件的顶层调用

第一个关键点:useState 是一个 Hook,因此你只能在 组件的顶层 或自己的 Hook 中调用它。你不能在循环或条件语句中调用它。如果你需要这样做,请提取一个新组件并将状态移入其中。

因为 hooks 为了在函数组件中引入状态,维护了一个有序表。

比如第一次执行函数组件时,我们拿到状态 count(通过useState,初始值为 0 )和 bool(通过 useState,初始值为 false),它们其实被保存到一个有序表中,它们的值会记录下来: [0, false]。

第二次执行函数组件, 会 按顺序 从这个表中拿出 0 和false,赋值给 count 和 bool。

如果你把 hook 写到判断条件下,导致某个 useState 不执行了,这里我们假设 count 的 useState 因为判断条件没有执行,会发生什么?结果是 bool 拿到了 0,发生了错位。

函数本身不能保存状态,我们需要额外维护一个有序的表,在执行 useState 之类的 hook 时,将它们保存到这个表里。

这要求每次函数组件的 hook 执行的位置相同,数量正确,否则会导致错位,不能拿到预期的状态值。

2.2 严格模式下的多次执行

第二个关键点:在严格模式中,React 将 两次调用组件函数、初始化函数和更新函数,以 帮你找到意外的不纯性。这只是开发时的行为,不影响生产。如果你的初始化函数是纯函数(本该是这样),就不应影响该行为。其中一个调用的结果将被忽略。

调用setState只会将更新函数放入队列,并不会立即修改当前的state,直到下次渲染的时候,state才会修改为更新后的值。那么如果你的setState是纯函数,在两次渲染之间无论执行多少次,都会有相同的state。反之,如果你的setState不纯,直接修改了当前的state,那么多次执行后的结果都不同。React使用Object.is 比较两个state是否相同。

下面是一个纯函数的例子:

function TodoList() {// 该函数组件会在每次渲染运行两次。const [todos, setTodos] = useState(() => {// 该初始化函数在初始化期间会运行两次。return createTodos();});function handleClick() {setTodos(prevTodos => {// 该更新函数在每次点击中都会运行两次return [...prevTodos, createTodo()];});}// ...

更新函数不纯的例子:

setTodos(prevTodos => {// 🚩 错误:改变 stateprevTodos.push(createTodo());
});

参考:

useState – React 中文文档

为什么 hooks 不能写在循环或者条件判断语句里_hooks为什么不能写在条件语句中-CSDN博客

相关文章:

  • 景联文科技:驾驭数据浪潮,赋能AI产业——全球领先的数据标注解决方案供应商
  • 新版本!飞凌嵌入式RK3568系列开发板全面支持Debian 11系统
  • C#开发的OpenRA游戏之世界存在的属性RenderDebugState(5)
  • 分类预测 | Matlab实现PSO-GRU粒子群算法优化门控循环单元的数据多输入分类预测
  • 使用Python轻松实现科研绘图
  • Ubuntu中安装R语言环境并在jupyter kernel里面增加R kernel
  • linux版:TensorRT安装教程
  • 基于PHP的化妆品销售网站,MySQL数据库,PHPstudy,前台用户+后台管理,完美运行,有一万多字论文
  • flink的副输出sideoutput单元测试
  • 基于opencv+tensorflow+神经网络的智能银行卡卡号识别系统——深度学习算法应用(含python、模型源码)+数据集(二)
  • js中的instance,isPrototype和getPrototypeOf的使用,来判断类的关系
  • 这就是思维导图!全面分析思维导图的实际用途
  • 深度学习_13_YOLO_图片切片及维度复原
  • 如何下载 Apache + PHP + Mysql 集成安装环境并结合内网穿透工具实现公网访问内网服务
  • 【 云原生 | K8S 】kubeadm 部署Kubernetes集群
  • 【翻译】babel对TC39装饰器草案的实现
  • canvas 绘制双线技巧
  • EOS是什么
  • JAVA之继承和多态
  • JS基础之数据类型、对象、原型、原型链、继承
  • Less 日常用法
  • SpiderData 2019年2月13日 DApp数据排行榜
  • 阿里云前端周刊 - 第 26 期
  • 分布式熔断降级平台aegis
  • 官方新出的 Kotlin 扩展库 KTX,到底帮你干了什么?
  • 经典排序算法及其 Java 实现
  • 精彩代码 vue.js
  • 深度学习入门:10门免费线上课程推荐
  • 消息队列系列二(IOT中消息队列的应用)
  • 回归生活:清理微信公众号
  • 直播平台建设千万不要忘记流媒体服务器的存在 ...
  • ​软考-高级-信息系统项目管理师教程 第四版【第14章-项目沟通管理-思维导图】​
  • $.proxy和$.extend
  • (4)logging(日志模块)
  • (C语言)深入理解指针2之野指针与传值与传址与assert断言
  • (react踩过的坑)antd 如何同时获取一个select 的value和 label值
  • (二)pulsar安装在独立的docker中,python测试
  • (图)IntelliTrace Tools 跟踪云端程序
  • (心得)获取一个数二进制序列中所有的偶数位和奇数位, 分别输出二进制序列。
  • (转)c++ std::pair 与 std::make
  • .NET6实现破解Modbus poll点表配置文件
  • /bin/rm: 参数列表过长"的解决办法
  • @Autowired @Resource @Qualifier的区别
  • @NestedConfigurationProperty 注解用法
  • @Validated和@Valid校验参数区别
  • [ 渗透测试面试篇 ] 渗透测试面试题大集合(详解)(十)RCE (远程代码/命令执行漏洞)相关面试题
  • [ 云计算 | AWS ] AI 编程助手新势力 Amazon CodeWhisperer:优势功能及实用技巧
  • [16/N]论得趣
  • [BZOJ1053][HAOI2007]反素数ant
  • [C++]C++类基本语法
  • [C++]指针与结构体
  • [Firefly-Linux] RK3568 pca9555芯片驱动详解
  • [Flutter]设置应用包名、名称、版本号、最低支持版本、Icon、启动页以及环境判断、平台判断和打包
  • [Godot] 3D拾取
  • [one_demo_16]直接插入排序的demo