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

React@16.x(29)useRef

目录

  • 1,介绍
  • 2,和 React.createRef() 的区别
  • 3,计时器的问题

目前来说,因为函数组件每次触发更新时,都会重新运行。无法像类组件一样让一些内容保持不变。
所以才出现了各种 HOOK 函数:useStateuseCallbackuseMemo 等来辅助实现和类组件相似的功能。

useRef 也是这样的目的。

1,介绍

在之前的文章中介绍了 ref,用于获取组件或真实DOM元素的引用。

useRef 作用相同,不过可以在函数组件中使用。同时它会返回对象的固定引用。

换句话说,当函数组件重新运行时,useRef() 前后2次返回的对象引用地址相同。
一个节点(React元素)对应一个唯一的对象。

React.createRef() 使用举例:

import React, { useState } from "react";export default function App() {const refInput = React.createRef();return (<><input ref={refInput}></input><buttononClick={() => {console.log(refInput.current.value);}}>获取 inputRef</button></>);
}

换成 useRef仅需要替换一行代码:

const refInput = React.createRef();
// 替换为
const refInput = useRef();

2,和 React.createRef() 的区别

上面的代码中,看起来只是一行代码的区别,但本质上处理逻辑不同。

  • React.createRef(),如果 ref 的值发生变动(函数组件重新渲染),则将旧值设为 null
  • useRef(),函数组件重新渲染多次时,所有返回的对象的引用地址相同。

验证下:
简单修改下,将每次更新后的新 ref 放到 window 对象中对比下:

import React, { useRef, useState } from "react";window.arr = [];
export default function App() {const refInput = React.createRef();window.arr.push(refInput);const [n, setN] = useState(); // 只是为了重新渲染组件。return (<><inputref={refInput}type="text"value={n}onChange={(e) => {setN(e.target.value);}}/></>);
}

多次改变 input.value 时,检查 window.arr

在这里插入图片描述

替换为 const refInput = useRef();

在这里插入图片描述

3,计时器的问题

在之前介绍 useEffect 时,提到了下面的写法是有问题的。

因为 useEffect 只会执行一次,所以在计时器中通过闭包获取的状态变量 n 永远都是10,

export default function App() {const [n, setN] = useState(10);useEffect(() => {const timer = setInterval(() => {setN(n - 1);}, 1000);return () => {clearInterval(timer);};}, []);return <div>{n}</div>;
}

该问题,通过将依赖项 n 传入即可。但会引起另一个问题:
每次函数重新运行,都会再次执行 useEffect,开启计时器又销毁计时器,开销很大。

export default function App() {const [n, setN] = useState(10);useEffect(() => {const timer = setInterval(() => {setN(n - 1);}, 1000);return () => {clearInterval(timer);};}, [n]);return <div>{n}</div>;
}

所以,可通过 useRef 来将函数重新运行后的新值传递给计时器,同时 useEffect 也只会运行一次,开启一次计时器!

注意,useRef() 返回的虽然是同一个对象,但 setN 修改的是它的 current 属性。所以计时器每次获取的都是新值。

export default function App() {const [n, setN] = useState(10);const refN = useRef(n);useEffect(() => {const timer = setInterval(() => {setN(--refN.current);}, 1000);return () => {clearInterval(timer);};}, []);return <div>{n}</div>;
}

以上。

http://www.lryc.cn/news/373468.html

相关文章:

  • 无人机的力量——在民用方面的应用
  • 探索档案未来,尽在ARCHE-2024
  • Maven 核心插件 maven-clean-plugin 使用详解
  • 金融数据中心布线运维管理解决方案
  • C++初学者指南第一步---2. Hello world
  • gitLab批量下载有权限的项目
  • 解决 kali 中使用 vulhub 拉取不到镜像问题
  • CSS3 简介
  • springboot事务管理的机制是什么
  • Linux下tar命令解压缩
  • 当财政支持减弱时,国有企业如何实现降本增效?
  • 大模型「训练」与「微调」概念详解【6000字长文】
  • JVM 垃圾回收器
  • Spring IOC 容器的构建流程?
  • 官方文档 搬运 MAXMIND IP定位 mysql导入 简单使用
  • PHP入门教程1:PHP的基础概念和基本语法
  • 头歌资源库(5)求阶乘问题
  • 09:整型与布尔型的转换
  • 51单片机STC89C52RC——2.1 独立按键控制LED亮灭
  • 系统架构师考点--计算机硬件
  • vite-plugin-mock前端自行模拟接口返回数据的插件
  • 网络安全知识全景地图V1.0 - 20240616更新
  • 力扣19. 删除链表的倒数第N个节点
  • 电脑找不到vcruntime140_1.dll的原因分析及5种解决方法分享
  • 洗地机哪个牌子质量好,性价比高?一文盘点市场热门选择
  • MySQL 之 JSON 支持(三)—— JSON 函数
  • 《华为项目管理之道》第1章笔记
  • C# —— 算数运算符
  • 去掉eslint
  • 【代码随想录算法训练Day38】LeetCode 509.斐波纳契数、LeetCode 76.爬楼梯、LeetCode 746. 使用最小花费爬楼梯