React 是一个流行的 JavaScript 库,用于构建用户界面。它提供了几个钩子,使开发人员能够管理状态并执行副作用。 React 中两个常用的钩子是 useRef 和 useState 。虽然它们乍一看似乎很相似,但它们具有不同的目的并且具有不同的用例。在本文中,我们将深入探讨 useRef 和 useState ,比较它们的功能并提供示例来说明它们的用法。

理解 useRef :

React 中的 useRef 钩子创建了一个在组件呈现之间持续存在的可变引用。与管理状态并触发重新渲染的 useState 不同, useRef 主要用于访问和操作 DOM 或存储不触发重新渲染的可变值。它返回一个带有 current 属性的可变对象。

示例 1:访问 DOM 元素

假设我们想在单击按钮时关注输入字段。我们可以使用 useRef 来实现这一点,如下所示:

import React, { useRef } from 'react';

function MyComponent() {
const inputRef = useRef(null); const handleClick = () => {
inputRef.current.focus();
}; return (
<div>
<input ref={inputRef} />
<button onClick={handleClick}>Focus Input</button>
</div>
);
}

在上面的示例中,我们使用 useRef 创建一个 ref 并将其分配给 inputRef 变量。我们将 inputRef 传递给输入元素的 ref 属性,使其可用于访问输入的 DOM 节点。单击按钮时,将执行 handleClick 函数,并调用 inputRef.current.focus() 以聚焦于输入字段。

理解 useState :

useState 挂钩用于管理功能组件内的状态。它允许我们创建可以更新的变量,并在其值发生变化时触发重新渲染。 useState 钩子返回一个包含两个元素的数组:当前状态值和更新它的函数。

示例 2:管理计数器

让我们使用 useState 创建一个简单的计数器组件:

import React, { useState } from 'react';

function Counter() {
const [count, setCount] = useState(0); const increment = () => {
setCount(count + 1);
}; return (
<div>
<p>Count: {count}</p>
<button onClick={increment}>Increment</button>
</div>
);
}

在上面的代码中,我们使用数组解构语法来分配 count 状态变量和 setCount 函数来更新它。使用 useState(0) 将 count 的初始值设置为 0 。单击按钮时,将调用 increment 函数,通过添加 1 更新 count 状态。结果,组件重新渲染,反映 count 的更新值。

比较 useRef 和 useState :

虽然 useRef 和 useState 都可以存储值,但它们有不同的用途:

  1. 管理状态: useState 旨在管理组件内的状态。当状态更新时,它会触发重新渲染,确保 UI 反映最新值。
  2. 访问和操作 DOM: useRef 主要用于与 DOM 交互,例如访问输入值或关注元素。它允许我们存储对 DOM 节点的引用并检索它们的属性,而无需触发重新渲染。
  3. 跨渲染保留值: useRef 在组件渲染之间维护相同的值,而 useState 在每次渲染期间初始化状态。
  4. 重新渲染行为:更新 useState 返回的值会导致组件重新渲染,同时更新使用 useRef 的 current 属性 不会触发重新渲染。

用例:

为了进一步了解 useRef 和 useState 的用例,让我们探讨一下每个钩子更适合的一些场景:

1. useRef 用例:

1.1.  访问 DOM 元素:当您需要访问或操作 DOM 元素(例如聚焦输入、滚动到特定元素或测量元素的大小)时, useRef 是合适的选择。它允许您创建对 DOM 节点的引用并访问其属性或方法。

1.2.  存储可变值:如果您有一个值需要在渲染过程中保留,但不会影响组件的 UI 或触发重新渲染, useRef 是一个不错的选择。例如,您可以使用 useRef 存储以前的值、缓存值或保留可变值以进行比较。

2. useState 用例:

2.1. 管理组件状态:当您需要管理和更新组件内的状态时,建议使用 useState 方法。它提供了一种存储和更新影响组件 UI 并触发重新渲染的值的方法。

2.2. 处理用户交互:如果组件中有交互元素(例如复选框、输入字段或切换开关),则通常使用 useState 来管理与这些交互相关的状态。您可以根据用户输入更新状态并反映 UI 中的更改。

对比示例:

为了更清楚地说明 useRef 和 useState 之间的区别,让我们考虑一个可以使用两个钩子的示例:

假设我们有一个带有输入字段和提交按钮的表单。当用户单击提交按钮时,我们希望在不清除输入字段的情况下显示成功消息。

使用 useRef

import React, { useRef } from 'react';

function Form() {
const inputRef = useRef(null); const handleSubmit = () => {
const value = inputRef.current.value;
// 提交表单
displaySuccessMessage();
}; const displaySuccessMessage = () => {
// 显示成功消息而不清除输入字段
}; return (
<div>
<input ref={inputRef} />
<button onClick={handleSubmit}>Submit</button>
</div>
);
}

在此示例中,我们使用 useRef 创建对输入字段的引用。单击提交按钮后,我们使用 inputRef.current.value 访问输入字段的值并继续提交表单。输入字段的值未清除,因为我们没有更新状态或触发重新渲染。

使用 useState :

import React, { useState } from 'react';

function Form() {
const [inputValue, setInputValue] = useState(''); const handleSubmit = () => {
// 提交表单
displaySuccessMessage();
}; const displaySuccessMessage = () => {
// 展示成功消息
setInputValue(''); // 清理输入内容
}; const handleInputChange = (e) => {
setInputValue(e.target.value);
}; return (
<div>
<input value={inputValue} onChange={handleInputChange} />
<button onClick={handleSubmit}>Submit</button>
</div>
);
}

在这个版本中,我们使用 useState 来管理输入字段的状态。我们使用 useState('') 用空字符串初始化 inputValue 状态。当用户在输入字段中键入内容时,将调用 handleInputChange 函数,更新状态并触发重新渲染以反映新值。单击提交按钮时,将执行 handleSubmit 函数,该函数显示成功消息并通过将 inputValue 状态设置为空字符串来清除输入字段。

在此示例中, useState 用于管理输入字段的值并在用户与其交互时触发重新渲染。 displaySuccessMessage 中的状态更新通过更新 inputValue 状态来清除输入字段。

结论:

总之, useRef 和 useState 都是React中必不可少的钩子,但它们有不同的用途。 useRef 主要用于访问和操作 DOM 或存储可变值而不触发重新渲染。它提供了一个在组件呈现之间持续存在的可变引用。另一方面, useState 用于管理组件状态,当状态更新时触发重新渲染。它返回一个状态值和一个更新它的函数。

了解 useRef 和 useState 之间的差异并了解何时使用每个钩子对于编写有效且优化的 React 组件至关重要。通过正确利用 useRef 和 useState ,您可以使用 React 构建交互式和高性能应用程序。

谢谢阅读!

我希望您觉得这篇文章有用。如果您有任何疑问或建议,请留言。您的反馈帮助我变得更好。

React 中的 useRef 与 useState的更多相关文章

  1. 用React 中的useState改变值不重新渲染的问题

    不渲染 const [lists,setLists] =useState([]); ..... const arr = lists; arr.splice(index,1) //根据删除index下标 ...

  2. 在React中使用 react-router-dom 编程式路由导航的正确姿势【含V5.x、V6.x】

    ## react-router-dom 编程式路由导航 (v5) ###### 1.push跳转+携带params参数 ```jsx props.history.push(`/b/child1/${i ...

  3. React 中阻止事件冒泡的问题

    在正式开始前,先来看看 JS 中事件的触发与事件处理器的执行. JS 中事件的监听与处理 事件捕获与冒泡 DOM 事件会先后经历 捕获 与 冒泡 两个阶段.捕获即事件沿着 DOM 树由上往下传递,到达 ...

  4. 哪种方式更适合在React中获取数据?

    作者:Dmitri Pavlutin 译者:小维FE 原文:dmitripavlutin.com 国外文章,笔者采用意译的方式,以保证文章的可读性. 当执行像数据获取这样的I/O操作时,你必须发起获取 ...

  5. 解读vue-server-renderer源码并在react中的实现

    前言 ​ 在博客开发的过程中,有这样一个需求想解决,就是在SSR开发环境中,服务端的代码是是直接通过webpack打包成文件(因为里面包含同构的代码,就是服务端与客户端共享前端的组件代码),写到磁盘里 ...

  6. 前端必读2.0:如何在React 中使用SpreadJS导入和导出 Excel 文件

    最近我们公司接到一个客户的需求,要求为正在开发的项目加个功能.项目的前端使用的是React,客户想添加具备Excel 导入/导出功能的电子表格模块. 经过几个小时的原型构建后,技术团队确认所有客户需求 ...

  7. react中redux怎么使用

    一.redux是什么? redux 就是react 全局状态管理,作用是存放全局数据 二.核心 state:存放数据 reducer:修改仓库数据 是一个函数,参数一:仓库中的数据,参数2:行为 ac ...

  8. 在react中使用wangEditorV5

    wangEditor是基于JavaScript和css的一款web富文本编辑器,是国内比较好用的一款轻量级富文本编辑器,上手简单,易用且开源免费. 官方文档:http://www.wangeditor ...

  9. 理解React中es6方法创建组件的this

    首发于:https://mingjiezhang.github.io/(转载请说明此出处). 在JavaScript中,this对象是运行时基于函数的执行环境(也就是上下文)绑定的. 从react中的 ...

  10. 【原】React中,map出来的元素添加事件无法使用

    在使用react中,经常用到react的map函数,用法和jquery里中的map一样,但是,如果你在每个map出来的元素中添加,你会发觉添加的事件无法关联, 比如,我们很多的评论,我需要在每个评论下 ...

随机推荐

  1. 微信小程序为什么引入 rpx

    前言 众所周知,px 是一个叫做像素的东西,pixel. 像素是指由图像的小方格组成的,这些小方块都有一个明确的位置和被分配的色彩数值,小方格颜色和位置就决定该图像所呈现出来的样子. 可以将像素视为整 ...

  2. MRBS(Meeting Room Booking System)开源的会议室预订系统搭建使用

    前一家公司所有的办公系统都是自己开发的,包括排班.工单.会议室预定等等,很方便. 目前所在的公司,每周部门例会找行政预订了会议室,但多次去都被人占了,很烦,于是网上找了一个评价不错的系统,python ...

  3. GaussDB(DWS)基于Flink的实时数仓构建

    本文分享自华为云社区<GaussDB(DWS)基于Flink的实时数仓构建>,作者:胡辣汤. 大数据时代,厂商对实时数据分析的诉求越来越强烈,数据分析时效从T+1时效趋向于T+0时效,为了 ...

  4. Javscript数组的常用方法有哪些?

    数组基本操作可以归纳为 增.删.改.查,需要留意的是哪些方法会对原数组产生影响,哪些方法不会 下面对数组常用的操作方法做一个归纳 增 下面前三种是对原数组产生影响的增添方法,第四种则不会对原数组产生影 ...

  5. 跨域是什么?Vue项目中你是如何解决跨域的呢?

    一.跨域是什么 跨域本质是浏览器基于同源策略的一种安全手段 同源策略(Sameoriginpolicy),是一种约定,它是浏览器最核心也最基本的安全功能 所谓同源(即指在同一个域)具有以下三个相同点 ...

  6. uniapp小程序页面实现元素与胶囊进行居中对齐

    无论是否为uni,关键在于获取胶囊中点的位置,如果是原生小程序根据小程序文档获取,其余逻辑处理是一致的 代码语法都只是技术选择,重点是逻辑处理,对于技术的运用,代码技术好比是积木,好的程序就是好的组合 ...

  7. 力扣162(java&python)-寻找峰值(中等)

    题目: 峰值元素是指其值严格大于左右相邻值的元素. 给你一个整数数组 nums,找到峰值元素并返回其索引.数组可能包含多个峰值,在这种情况下,返回 任何一个峰值 所在位置即可. 你可以假设 nums[ ...

  8. 力扣1235(java)-规划兼职工作(困难)

    题目: 你打算利用空闲时间来做兼职工作赚些零花钱. 这里有 n 份兼职工作,每份工作预计从 startTime[i] 开始到 endTime[i] 结束,报酬为 profit[i]. 给你一份兼职工作 ...

  9. RocketMQ 之 IoT 消息解析:物联网需要什么样的消息技术?

    前言: 从初代开源消息队列崛起,到 PC 互联网.移动互联网爆发式发展,再到如今 IoT.云计算.云原生引领了新的技术趋势,消息中间件的发展已经走过了 30 多个年头. 目前,消息中间件在国内许多行业 ...

  10. DNS高可用设计--软件高可用

    DNS是网络的基础服务,网络上的各种应用对DNS的依赖性很高.DNS的稳定,直接决定了上层应用服务的稳定.那如何保障DNS服务的高可用呢?我们先来看下高可用的概念: 高可用 高可用(High avai ...