0

0

JS如何实现useRef?Ref的持久化

月夜之吻

月夜之吻

发布时间:2025-08-23 12:41:01

|

1006人浏览过

|

来源于php中文网

原创

useRef能持久化是因为它返回的对象在组件实例的生命周期内始终保持同一引用,React通过将该对象绑定到组件的内部节点(如Fiber节点)实现跨渲染的持久存储,每次调用useRef都返回同一实例,确保.current值在多次渲染间不变且修改不触发重渲染。

js如何实现useref?ref的持久化

useRef
在JavaScript(尤其是在React这类前端框架的语境下)的实现,核心在于它提供了一个在组件多次渲染之间保持不变的、可变的容器。它不是通过什么复杂的响应式系统来追踪变化,而仅仅是给你一个普通的对象,这个对象在组件的整个生命周期内都是同一个实例,并且它的
.current
属性可以被随意修改,而不会触发组件的重新渲染。你可以把它想象成一个组件内部的“私有储物柜”,你放进去的东西,下次渲染时还在那里,不会因为组件更新而消失或重置。

本质上,

useRef
的核心在于它返回一个普通JavaScript对象,这个对象拥有一个
current
属性。关键在于,无论组件重新渲染多少次,
useRef
每次都会返回同一个对象实例
。它不是通过什么魔法实现持久化的,而是在React(或其他类似框架)的内部机制中,将这个特定的引用对象与组件的特定实例(或者说,与组件在渲染树中的位置)绑定起来了。当你第一次调用
useRef(initialValue)
时,框架会创建一个
{ current: initialValue }
这样的对象并存储起来。后续的渲染中,当同一个组件再次调用
useRef
时,框架会检查是否已经为这个组件实例存储了一个引用对象,如果有,就直接返回那个已经存在的对象,而不是重新创建一个。这确保了
current
属性即使在组件外部被修改,其状态也能在不同渲染周期中保持一致,且不会触发组件重新渲染。

useRef
为什么能持久化?它的内部机制是怎样的?

useRef
能够持久化的秘密,其实藏在前端框架(比如React)的内部调度和状态管理机制里。当一个组件首次被渲染时,框架会为它创建一个内部的“实例”或者说“纤维(Fiber)”节点。这个节点就像是组件在内存中的一个代表,它会存储组件的所有内部状态、副作用以及,没错,就是
useRef
创建的那些引用对象。

当你第一次在组件里调用

useRef(initialValue)
时,React会识别到这个调用,然后它会为当前的组件实例生成一个形如
{ current: initialValue }
的普通JavaScript对象,并把这个对象“挂载”到组件对应的那个内部节点上。你可以把这个内部节点想象成一个组件的“背包”,
useRef
创建的对象就被放进了这个背包里。

接下来,无论你的组件因为什么原因(比如父组件重渲染、自身状态改变)而重新渲染,React在执行组件函数时,会再次遇到

useRef
的调用。这时候,React并不会傻乎乎地再创建一个新对象。它会聪明地检查当前组件的“背包”里,是不是已经有一个之前存进去的
useRef
对象了。如果找到了,它就会直接把那个旧的、已经存在的对象返回给你。

正是这种“首次创建,后续复用”的策略,确保了

useRef
返回的永远是同一个对象实例。它的
current
属性虽然可以被修改,但这种修改并不会被React的响应式系统追踪,所以它不会像
useState
那样触发组件的重新渲染。这就像你有一个私人的笔记本,你在上面写写画画,但没人会因此而大声宣布“笔记本内容变了,快来看!”。

useRef
useState
有什么本质区别?何时应该选择它们?

这俩兄弟在表面上看起来都是用来“存东西”的,但骨子里却完全不同,用错了地方可能就是灾难。

useState
是用来管理组件的“状态”的。这里的“状态”指的是那些会影响组件渲染结果、需要被UI响应式地追踪和更新的数据。当你通过
useState
更新一个值时,React会知道这个值变了,然后它会重新渲染组件,让UI反映出最新的状态。它就像一个公共的布告栏,上面贴着最新的信息,信息一变,大家就都知道了。

useRef
,正如前面所说,它提供的是一个“持久化的、可变的容器”。它的核心特点是:修改
.current
属性不会触发组件重新渲染。它更像是一个组件内部的“私人物品”,你可以在里面放任何东西,比如DOM元素的引用、一个计时器ID、一个WebSocket实例,或者任何你希望在多次渲染之间保持不变但又不需要触发UI更新的数据。

选择时机:

  • 选择

    useState

    零沫AI工具导航
    零沫AI工具导航

    零沫AI工具导航-AI导航新标杆,探索全球实用AI工具

    下载
    • 当你的数据变化需要直接反映在用户界面上时(比如用户输入、数据加载状态、一个开关的开启/关闭)。
    • 你需要通过某种方式(比如表单提交、按钮点击)来“设置”这个值,并且希望UI能立即响应。
    • 数据是组件渲染逻辑的直接输入。
  • 选择

    useRef

    • 你需要直接操作DOM元素(比如获取输入框的焦点、播放/暂停视频)。
    • 你需要存储一个在组件生命周期内保持不变的引用,但这个引用值的改变不应该触发组件重新渲染(比如计时器ID、第三方库的实例、WebSocket连接)。
    • 你需要在多次渲染之间“记住”某个值,但这个值本身不是UI的状态(比如上一次渲染的值、一个计算结果的缓存)。
    • 你想要存储一个可变的对象,并且希望在不触发重渲染的情况下修改它。

简单来说,如果数据的变化需要“告诉”用户界面,那就用

useState
;如果数据只是组件内部的“私事”,或者它是一个需要直接操作的“引用”,那就用
useRef
。我个人觉得,很多时候新手容易混淆,但只要记住“是否需要触发重渲染”这个核心判断点,就能少走很多弯路。

除了DOM引用,
useRef
还有哪些不为人知的妙用?

useRef
的功能远不止于获取DOM引用,它的“持久化”特性让它在很多场景下都非常有用,甚至能解决一些看似棘手的问题。

一个很常见的场景是存储上一次的值(Previous Value)。假设你想知道一个

prop
或者
state
在当前渲染周期中,它的上一个值是什么。你不能直接用一个普通变量去存,因为普通变量在每次组件渲染时都会重新初始化。这时候
useRef
就派上用场了:

function MyComponent({ value }) {
  const prevValueRef = useRef(); // 创建一个ref来存储上一个值

  useEffect(() => {
    // 在effect中更新ref的current属性,因为它会在渲染之后执行
    prevValueRef.current = value;
  }, [value]); // 只有当value变化时才执行

  // 现在你可以随时访问 prevValueRef.current 来获取上一个值了
  const prevValue = prevValueRef.current;

  // ... 你的组件逻辑
  return (
    <div>
      <p>当前值: {value}</p>
      <p>上一个值: {prevValue !== undefined ? prevValue : '无'}</p>
    </div>
  );
}

这种模式在处理动画、数据比较或者某些复杂的副作用逻辑时非常有用。

另一个妙用是存储可变的实例或计时器ID。比如,你可能需要在组件挂载时启动一个

setInterval
,在组件卸载时清除它。
setInterval
返回的ID需要被“记住”,以便在清理函数中引用。

function TimerComponent() {
  const intervalRef = useRef(null); // 用ref来存储计时器ID

  useEffect(() => {
    intervalRef.current = setInterval(() => {
      console.log('计时器在运行...');
    }, 1000);

    return () => {
      // 组件卸载时清除计时器
      if (intervalRef.current) {
        clearInterval(intervalRef.current);
      }
    };
  }, []); // 空依赖数组表示只在挂载和卸载时执行

  return <div>查看控制台,我正在计时...</div>;
}

这种方式确保了计时器ID在组件的整个生命周期内都是可访问的,并且可以在清理函数中正确使用。

我个人还喜欢用

useRef
存储一些不需要响应式更新的“状态”或“标志”。比如,你可能想知道一个组件是否已经挂载(
isMounted
标志),或者一个异步操作是否正在进行中,但这些信息并不需要直接触发UI更新,只是作为内部逻辑判断的依据。

function AsyncDataFetcher() {
  const isMounted = useRef(false); // 存储组件是否挂载的标志

  useEffect(() => {
    isMounted.current = true; // 组件挂载时设置为true

    // 模拟数据请求
    const fetchData = async () => {
      // ... 假设这里有一些异步操作
      await new Promise(resolve => setTimeout(resolve, 2000));
      if (isMounted.current) { // 在更新状态前检查组件是否仍然挂载
        console.log("数据已获取,组件仍在挂载");
        // setState(...)
      } else {
        console.log("数据已获取,但组件已卸载,不再更新状态");
      }
    };

    fetchData();

    return () => {
      isMounted.current = false; // 组件卸载时设置为false
    };
  }, []);

  return <div>模拟异步数据获取...</div>;
}

这个

isMounted
的模式在处理异步操作时特别有用,可以避免在组件已经卸载后尝试更新状态,从而防止内存泄漏或意外行为。它提供了一种简洁而有效的方式来管理组件的生命周期状态,而无需引入额外的
useState
和不必要的渲染。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

531

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

576

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

760

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

6258

2023.08.17

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

492

2023.09.01

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

Js中concat和push的区别
Js中concat和push的区别

Js中concat和push的区别:1、concat用于将两个或多个数组合并成一个新数组,并返回这个新数组,而push用于向数组的末尾添加一个或多个元素,并返回修改后的数组的新长度;2、concat不会修改原始数组,是创建新的数组,而push会修改原数组,将新元素添加到原数组的末尾等等。本专题为大家提供concat和push相关的文章、下载、课程内容,供大家免费下载体验。

240

2023.09.14

js截取字符串的方法介绍
js截取字符串的方法介绍

JavaScript字符串截取方法,包括substring、slice、substr、charAt和split方法。这些方法可以根据具体需求,灵活地截取字符串的不同部分。在实际开发中,根据具体情况选择合适的方法进行字符串截取,能够提高代码的效率和可读性 。

303

2023.09.21

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号