0

0

掌握 React useRef 中数组的过滤与更新:避免常见陷阱

花韻仙語

花韻仙語

发布时间:2025-10-27 12:54:11

|

653人浏览过

|

来源于php中文网

原创

掌握 React useRef 中数组的过滤与更新:避免常见陷阱

在使用 react `useref` 存储数组并尝试对其进行过滤时,一个常见误区是期望 `array.prototype.filter()` 方法能原地修改数组。实际上,`filter()` 会返回一个新数组,因此必须将这个新数组显式地重新赋值给 `ref.current` 才能实现更新。同时,访问 `ref` 中数组的属性(如 `length`)时,务必通过 `ref.current.length` 进行。

理解 useRef 与其在数组操作中的特性

useRef 是 React 提供的一个 Hook,它允许你在函数组件中存储一个可变的引用,这个引用在组件的整个生命周期中都会持续存在,并且更新它不会触发组件的重新渲染。这使得 useRef 非常适合存储那些不需要响应式更新但需要在不同渲染之间保持不变的数据,例如 DOM 元素的引用、计时器 ID,或者像本例中不直接用于渲染的数组。

当我们将一个数组存储在 useRef 中时,我们实际上是通过 ref.current 来访问和操作这个数组。例如:

let items = useRef([]); // 初始化一个空的数组引用
// ...
items.current = data; // 从数据库获取数据后赋值给引用

数组过滤的常见误区:filter() 的非原地修改特性

在 JavaScript 中,Array.prototype.filter() 方法是一个非原地修改(non-mutating)方法。这意味着它不会改变原始数组,而是根据提供的回调函数创建一个包含通过测试的新数组。

考虑以下代码片段:

items.current.filter((item) => item.name !== toy);

这段代码会正确地生成一个不包含指定 toy 的新数组。然而,它并没有将这个新数组重新赋值给 items.current。因此,items.current 仍然指向原始的、未经过滤的数组。这就是为什么在 console.log(items.current) 时会发现数组内容没有变化的原因。

正确更新 useRef 中数组的方法

要正确地从 useRef 存储的数组中移除元素,你需要将 filter() 方法返回的新数组重新赋值给 items.current。

艾绘
艾绘

艾绘:一站式绘本创作平台,AI智能绘本设计神器!

下载
// 假设 items 是通过 useRef([]) 定义的
// toy 是要移除的元素的名称
items.current = items.current.filter((item) => item.name !== toy);

通过这种方式,items.current 现在指向了经过过滤后的新数组,从而实现了对 useRef 中数组的“更新”。

访问 useRef 中数组属性的正确姿势

另一个常见的错误是直接尝试访问 ref 对象的属性,而不是其 current 属性所指向的值的属性。例如,在检查数组长度时:

// 错误示例:items 是一个 ref 对象,没有 length 属性
if (items.length === 0) {
  console.log('Winner');
  // ...
}

items 本身是一个 RefObject 类型,它只有一个 current 属性。要获取数组的长度,你必须通过 items.current 来访问数组,然后获取其 length 属性。

// 正确示例:通过 items.current 访问数组的 length 属性
if (items.current.length === 0) {
  console.log('Winner');
  navigate('/leaderboard', { state: time });
}

完整的修正示例

结合上述两点修正,原始 handleAction 函数的正确实现应如下:

function handleAction(click, toy) {
  const item = items.current.find((item) => item.name === toy);

  if (!item) {
    setFound(`Not quite, try again!`);
    return;
  }

  if (click.x > item.left && click.x < item.right) {
    if (click.y < item.bottom && click.y > item.top) {
      setFound(`Well done! You've found Sarah's ${toy}`);
      // 修正1:重新赋值过滤后的数组
      items.current = items.current.filter((i) => i.name !== toy);
      console.log(items.current);

      // 修正2:通过 items.current.length 检查数组长度
      if (items.current.length === 0) {
        console.log('Winner');
        navigate('/leaderboard', { state: time });
      }
    }
  } else {
    setFound(`Not quite, try again!`);
    return;
  }
}

总结与注意事项

  • filter() 不会原地修改数组:始终记住 Array.prototype.filter() 返回的是一个新数组。如果你想“更新”原始数组,必须将新数组重新赋值给变量。
  • useRef 的访问方式:useRef 存储的值总是通过 .current 属性来访问和修改。无论是读取还是写入,都必须使用 ref.current。
  • useRef 与 useState 的选择
    • 当数据变化不需要触发组件重新渲染时,使用 useRef。例如,本例中 items 数组的改变不直接影响 UI 元素的渲染,它只是一个内部状态管理。
    • 当数据变化需要触发组件重新渲染以更新 UI 时,使用 useState。如果 items 数组的改变直接导致页面上渲染的元素列表发生变化,那么它应该存储在 useState 中。
  • 可变性与不可变性:虽然 useRef 允许你存储可变数据,但在处理数组时,采用 filter() 这种不可变操作(创建新数组)并重新赋值,通常比 splice() 这种原地修改操作更安全、更易于理解和调试,尤其是在大型项目中。

通过理解 useRef 的工作原理以及 JavaScript 数组方法的特性,可以有效避免在 React 应用中处理可变数据时常见的陷阱。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
length函数用法
length函数用法

length函数用于返回指定字符串的字符数或字节数。可以用于计算字符串的长度,以便在查询和处理字符串数据时进行操作和判断。 需要注意的是length函数计算的是字符串的字符数,而不是字节数。对于多字节字符集,一个字符可能由多个字节组成。因此,length函数在计算字符串长度时会将多字节字符作为一个字符来计算。更多关于length函数的用法,大家可以阅读本专题下面的文章。

926

2023.09.19

console接口是干嘛的
console接口是干嘛的

console接口是一种用于在计算机命令行或浏览器开发工具中输出信息的工具,提供了一种简单的方式来记录和查看应用程序的输出结果和调试信息。本专题为大家提供console接口相关的各种文章、以及下载和课程。

415

2023.08.08

console.log是什么
console.log是什么

console.log 是 javascript 函数,用于在浏览器控制台中输出信息,便于调试和故障排除。想了解更多console.log的相关内容,可以阅读本专题下面的文章。

505

2024.05.29

DOM是什么意思
DOM是什么意思

dom的英文全称是documentobjectmodel,表示文件对象模型,是w3c组织推荐的处理可扩展置标语言的标准编程接口;dom是html文档的内存中对象表示,它提供了使用javascript与网页交互的方式。想了解更多的相关内容,可以阅读本专题下面的文章。

3335

2024.08.14

DOM是什么意思
DOM是什么意思

dom的英文全称是documentobjectmodel,表示文件对象模型,是w3c组织推荐的处理可扩展置标语言的标准编程接口;dom是html文档的内存中对象表示,它提供了使用javascript与网页交互的方式。想了解更多的相关内容,可以阅读本专题下面的文章。

3335

2024.08.14

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

167

2026.01.28

包子漫画在线官方入口大全
包子漫画在线官方入口大全

本合集汇总了包子漫画2026最新官方在线观看入口,涵盖备用域名、正版无广告链接及多端适配地址,助你畅享12700+高清漫画资源。阅读专题下面的文章了解更多详细内容。

35

2026.01.28

ao3中文版官网地址大全
ao3中文版官网地址大全

AO3最新中文版官网入口合集,汇总2026年主站及国内优化镜像链接,支持简体中文界面、无广告阅读与多设备同步。阅读专题下面的文章了解更多详细内容。

74

2026.01.28

php怎么写接口教程
php怎么写接口教程

本合集涵盖PHP接口开发基础、RESTful API设计、数据交互与安全处理等实用教程,助你快速掌握PHP接口编写技巧。阅读专题下面的文章了解更多详细内容。

2

2026.01.28

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 4.3万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

React核心原理新老生命周期精讲
React核心原理新老生命周期精讲

共12课时 | 1万人学习

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

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