带你了解React中的Ref,值得了解的知识点分享

本篇文章带大家了解一下react中的ref,介绍一些关于 ref 你需要知道的知识点,希望对大家有所帮助!

带你了解React中的Ref,值得了解的知识点分享

Intro

在 React 项目中,有很多场景需要用到 Ref。例如使用 ref 属性获取 DOM 节点,获取 ClassComponent 对象实例;用 useRef Hook 创建一个 Ref 对象,以便解决像 setInterval 获取不到最新的 state 的问题;你也可以调用 React.createRef 方法手动创建一个 Ref 对象。【相关推荐:Redis视频教程】

虽然 Ref 用起来也很简单,但在实际项目中实战还是难免遇到问题,这篇文章将从源码的角度出发梳理各种和 Ref 相关的问题,理清和 ref 相关的 API 背后都干了什么。看完这篇文章或许可以让你对的 Ref 有更深入地认识。

Ref 相关的类型声明

首先 refreference 的简称,也就是引用。在 react 的类型声明文件中,可以找到好几个和 Ref 相关的类型,这里将它们一一列举出来。

RefObject/MutableRefObject

interface RefObject { readonly current: T | null; }interface MutableRefObject { current: T; }

使用 useRef Hook 的时候返回的就是 RefObject/MutableRefObejct,这两个类型都是定义了一个 { current: T } 的对象结构,区别是 RefObject 的 current 属性是只读的,如果修改 refObject.current,Typescript 会警告⚠️。

const ref = useRef(null)ref.current = '' // Error

TS 报错:无法分配到 “current” ,因为它是只读属性。

1.png

查看 useRef 方法的定义,这里用了函数重载,当传入的泛型参数 T 不包含 null 时返回RefObject,当包含 null 时将返回 MutableRefObject

function useRef(initialValue: T): MutableRefObject;function useRef(initialValue: T | null): RefObject;

所以如果你希望创建的 ref 对象 current 属性是可修改的,需要加上 | null

const ref = useRef(null)ref.current = '' // OK

调用 React.createRef() 方法时返回的也是一个 RefObject

createRef

export function createRef(): RefObject {  const refObject = {    current: null,  };  if (__DEV__) {    Object.seal(refObject);  }  return refObject;}

RefObject/MutableRefObject 是在 16.3 版本才新增的,如果使用更早的版本,需要使用 Ref Callback

RefCallback

使用 Ref Callback 就是传递一个回调函数,react 回调时会将对应的实例回传过来,可以自行保存以便调用。这个回调函数的类型就是 RefCallback

type RefCallback = (instance: T | null) => void;

使用 RefCallback 示例:

import React from 'react'export class CustomTextInput extends React.Component {  textInput: HTMLInputElement | null = null;  saveInputRef = (element: HTMLInputElement | null) => {    this.textInput = element;  }  render() {    return (          );  }}

Ref/LegacyRef

在类型声明中,还有 Ref/LegacyRef 类型,它们用于泛指 Ref 类型。 LegacyRef 是兼容版本,在之前的老版本 ref 还可以是 字符串。

type Ref = RefCallback | RefObject | null;type LegacyRef = string | Ref;

理解了和 Ref 相关的类型,写起 Typescript 来才能更得心应手。

Ref 的传递

特殊的 props

在 JSX 组件上使用 ref 时,我们是通过给 ref 属性设置一个 Ref。我们都知道 jsx 的语法,会被 Babel 等工具编译成 createElement 的形式。

// jsx// compiled toReact.createElement(App, {  ref: ref,  id: "my-app"});

看起来 ref 和其他 prop 没啥区别,不过如果你尝试在组件内部打印 props.ref 却是 undefined。并且 dev 环境控制台会给出提示。

Trying to access it will result in undefined being returned. If you need to access the same value within the child component, you should pass it as a different prop.

React 对 ref 做了啥?在 ReactElement 源码中可以看到,refRESERVED_PROPS,同样有这种待遇的还有 key,它们都会被特殊处理,从 props 中提取出来传递给 Element

const RESERVED_PROPS = {  key: true,  ref: true,  __self: true,  __source: true,};

所以 ref 是会被特殊处理的 “props“

forwardRef

16.8.0 版本之前,Function Component 是无状态的,只会根据传入的 props render。有了 Hook 之后不仅可以有内部状态,还可以暴露方法供外部调用(需要借助 forwardRefuseImperativeHandle)。

如果直接对一个 Function Componentref,dev 环境下控制台会告警,提示你需要用 forwardRef 进行包裹起来。

function Input () {    return }const ref = useRef()

Function components cannot be given refs. Attempts to access this ref will fail. Did you mean to use React.forwardRef()?

forwardRef 为何物?查看源码 ReactForwardRef.js 将 __DEV__ 相关的代码折叠起来,它只是一个无比简单的高阶组件。接收一个 render 的 FunctionComponent,将它包裹一下定义 $$typeofREACT_FORWARD_REF_TYPEreturn 回去。

知了zKnown 知了zKnown

知了zKnown:致力于信息降噪 / 阅读提效的个人知识助手。

知了zKnown 65 查看详情 知了zKnown

2.png

跟踪代码,找到 resolveLazyComponentTag,在这里 $$typeof 会被解析成对应的 WorkTag。

3.png

REACT_FORWARD_REF_TYPE 对应的 WorkTag 是 ForwardRef。紧接着 ForwardRef 又会进入 updateForwardRef 的逻辑。

case ForwardRef: {  child = updateForwardRef(    null,    workInProgress,    Component,    resolvedProps,    renderLanes,  );  return child;}

这个方法又会调用 renderWithHooks 方法,并在第五个参数传入 ref

nextChildren = renderWithHooks(  current,  workInProgress,  render,  nextProps,  ref, // 这里  renderLanes,);

继续跟踪代码,进入 renderWithHooks 方法,可以看到,ref 会作为 Component 的第二个参数传递。到这里我们可以理解被 forwardRef 包裹的 FuncitonComponent 第二个参数 ref 是从哪里来的(对比 ClassComponent contructor 第二个参数是 Context)。

4.png

了解如何传递 ref,那下一个问题就是 ref 是如何被赋值的。

ref 的赋值

打断点(给 ref 赋值一个 RefCallback,在 callback 里面打断点) 跟踪到代码 createRef1,在这个方法里面,会判断 Fiber 节点的 ref 是 function 还是 RefObject,依据类型处理 instance。如果这个 Fiber 节点是 HostComponent (tag = 5) 也就是 DOM 节点,instance 就是该 DOM 节点;而如果该 Fiber 节点是 ClassComponent (tag = 1),instance 就是该对象实例。

function commitAttachRef(finishedWork) {  var ref = finishedWork.ref;  if (ref !== null) {    var instanceToUse = finishedWork.stateNode;    if (typeof ref === 'function') {      ref(instanceToUse);    } else {      ref.current = instanceToUse;    }  }}

以上是 HostComponent 和 ClassComponent 中对 ref 的赋值逻辑,对于 ForwardRef 类型的组件走的是另外的代码,但行为基本是一致的,可以看这里 createRef2。

接下里,我们继续挖掘 React 源码,看看 useRef 是如何实现的。

useRef 的内部实现

通过跟踪代码,定位到 useRef 运行时的代码 createRef3

5.png

这里有两个方法,mountRefupdateRef,顾名思义就是对应 Fiber 节点 mountupdate 时对 ref 的操作。

function updateRef(initialValue: T): {|current: T|} {  const hook = updateWorkInProgressHook();  return hook.memoizedState;}function mountRef(initialValue: T): {|current: T|} {  const hook = mountWorkInProgressHook();  const ref = {current: initialValue};  hook.memoizedState = ref;  return ref;}

可以看到 mount 时,useRef 创建了一个 RefObject,并将它赋值给 hookmemoizedStateupdate 时直接将它取出返回。

不同的 Hook memoizedState 保存的内容不一样,useState 中保存 state 信息, useEffect 中 保存着 effect 对象,useRef 中保存的是 ref 对象…

mountWorkInProgressHookupdateWorkInProgressHook 方法背后是一条 Hooks 的链表,在不修改链表的情况下,每次 render useRef 都能取回同一个 memoizedState 对象,就这么简单。

应用:合并 ref

至此,我们了解了在 React 中 ref 的传递和赋值逻辑,以及 useRef 相关的源码。用一个应用题来巩固以上知识点:有一个 Input 组件,在组件内部需要通过 innerRef HTMLInputElement 来访问 DOM 节点,同时也允许组件外部 ref 该节点,需要怎么实现?

const Input = forwardRef((props, ref) => {  const innerRef = useRef(null)  return (      )})

考虑一下上面代码中的 ??? 应该怎么写。

============ 答案分割线 ==============

通过了解 Ref 相关的内部实现,很明显我们这里可以创建一个 RefCallback,在里面对多个 ref 进行赋值就可以了。

export function combineRefs(  refs: Array<MutableRefObject | RefCallback>): React.RefCallback {  return value => {    refs.forEach(ref => {      if (typeof ref === 'function') {        ref(value);      } else if (ref !== null) {        ref.current = value;      }    });  };}const Input = forwardRef((props, ref) => {  const innerRef = useRef(null)  return (      )})

更多编程相关知识,请访问:createRef4!!

以上就是带你了解React中的Ref,值得了解的知识点分享的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/552611.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
Java框架助力企业级应用的高效开发
上一篇 2025年11月9日 20:46:27
笔记本重装系统蓝屏怎么解决
下一篇 2025年11月9日 20:46:31

相关推荐

  • React组件中动态属性值的管理与同步:利用状态实现受控组件

    本教程旨在解决react组件中动态属性值同步使用的问题。我们将探讨如何利用react的`usestate` hook来管理组件内部状态,从而实现一个属性的值动态地影响另一个属性,并构建出可预测、易于维护的受控组件。文章将通过具体代码示例,详细阐述从初始化状态到处理状态更新的完整过程,并强调受控组件在…

    2026年5月10日
    000
  • 基于两数组数据计算结果排序的 React 教程

    本教程针对 React 应用中需要根据两个独立数组的数据计算结果进行排序的场景,提供了一种高效的解决方案。通过使用 JavaScript 的 `reduce` 和 `map` 方法,将两个数组根据唯一标识符进行合并,从而简化排序逻辑,提高代码的可读性和可维护性。避免了复杂的嵌套循环或同步迭代,提供了…

    2026年5月10日
    000
  • 解决React中按钮点击不显示弹出表单的问题:状态管理与语法修正

    本教程旨在解决react应用中点击按钮后弹出表单未能正确渲染的问题。核心在于识别并修正代码中的语法错误以及未定义的react状态管理函数。我们将详细探讨如何使用`usestate`等react hooks来声明和管理组件状态,确保交互逻辑的正确实现,并提供结构清晰的代码示例,帮助开发者构建功能完善的…

    2026年5月10日
    000
  • Go语言与Microsoft SharePoint集成指南

    Go语言可以有效集成Microsoft SharePoint,主要通过两种途径:一是利用SharePoint提供的RESTful API进行数据交互,Go的标准HTTP客户端库即可轻松实现;二是通过SharePoint应用模型开发自托管应用,这种模型支持使用包括Go在内的任何语言编写后端逻辑。 1.…

    2026年5月10日
    000
  • javascript生命周期钩子是什么_组件有哪些关键阶段?

    JavaScript原生无生命周期钩子,这是Vue、React等框架为组件设计的机制;Vue按创建、挂载、更新、卸载四阶段提供对应钩子,React类组件有明确生命周期方法,函数组件则通过useEffect模拟,其核心价值在于精准控制执行时机以避免DOM操作错误和内存泄漏。 JavaScript 本身…

    2026年5月10日
    000
  • React Redux 中 useSelector 的自动订阅与取消订阅机制

    React Redux 中 useSelector 的自动订阅与取消订阅机制React Redux 中 useSelector 的自动订阅与取消订阅机制React Redux 中 useSelector 的自动订阅与取消订阅机制React Redux 中 useSelector 的自动订阅与取消订阅机制

    本文深入探讨 react redux 中 `useselector` hook 的核心机制。它详细解释了 `useselector` 如何在组件挂载时自动订阅 redux store 的状态更新,并在组件卸载时智能地取消订阅。这确保了应用程序的性能和内存效率,避免了对已卸载组件进行不必要的更新,从而…

    2026年5月10日 用户投稿
    100
  • 在 React 中实现用户输入停止检测的防抖策略

    本文详细介绍了在 React 应用中如何精确检测用户停止输入行为。通过引入防抖(Debounce)函数,可以有效优化输入事件处理,避免频繁触发不必要的网络请求或状态更新。文章提供了基于 React Hooks 的防抖实现示例,并探讨了其在提升用户体验和系统性能方面的应用,确保在用户停止输入指定时间后…

    用户投稿 2026年5月10日
    000
  • JavaScript中的标签模板字面量(Tagged Templates)有哪些高级用法?

    标签模板通过自定义函数实现复杂逻辑,如html函数转义防止XSS,css函数生成唯一类名封装样式,结合哈希值隔离组件样式,确保安全与模块化。 标签模板字面量不只是字符串拼接工具,它能结合函数实现更复杂的逻辑处理。通过自定义标签函数,你可以解析模板中的表达式和静态部分,从而实现如国际化、样式封装、安全…

    2026年5月10日
    000
  • 深入理解React组件命名规范:解决组件不渲染的常见陷阱

    本教程深入探讨react组件命名约定在组件渲染中的关键作用。我们将解释为何自定义组件名必须以大写字母开头(pascalcase),以避免与原生html元素混淆。通过对比错误和正确的代码示例,教程将指导开发者如何遵循这一核心规范,从而解决组件不显示、`is defined but never used…

    2026年5月10日
    000
  • 优化React-Redux应用中的用户与受保护数据按需加载

    本教程旨在解决React-Redux应用中用户数据和受保护API密钥在用户未登录时仍被请求,导致401错误的问题。通过引入条件性Redux状态初始化和动作分发逻辑,确保只有在用户被认为已认证时才发起相关的API请求,从而优化应用性能,减少不必要的网络流量和控制台错误。 在构建现代Web应用时,尤其是…

    2026年5月10日
    000
  • 全栈JS代码怎么结构化_全栈JavaScript项目代码结构与规范指南

    采用分层+功能划分的目录结构,明确分离前后端代码;2. 遵循单一职责原则,路由、控制器、服务与模型各司其职;3. 统一命名规范并集成ESLint+Prettier保证代码风格一致;4. 使用环境变量管理配置,通过脚本实现自动化构建与并发启动服务。 全栈JavaScript项目涉及前端、后端、数据库交…

    2026年5月10日
    000
  • JavaScript模块加载机制_JavaScript代码组织规范

    现代前端推荐使用ES Modules,通过import和export实现静态依赖管理,配合合理目录结构与命名规范提升可维护性,注意浏览器与Node.js的运行差异。 JavaScript 的模块加载机制和代码组织规范是现代前端开发中的核心基础。随着项目规模扩大,良好的模块化设计能提升代码可维护性、复…

    2026年5月10日
    000
  • 前端实现记住密码与自动填充_javascript技巧

    正确使用表单标签与属性、支持“记住我”功能、避免破坏自动填充机制、测试浏览器兼容性可实现稳定自动填充。1. 使用标准input类型并设置autocomplete属性为username和current-password;2. 登录成功后通过localStorage保存用户名,页面加载时恢复;3. 避免…

    2026年5月10日
    000
  • 深入理解React中Refs、DOM组件与类组件实例的Ref转发机制

    本文旨在澄清react中“dom组件”的概念,并深入探讨refs在原生dom元素和自定义组件(特别是类组件实例)之间的转发机制。我们将解析官方文档中的常见困惑,并通过示例代码演示如何正确地将refs转发给不同的组件类型,从而帮助开发者更好地利用refs进行dom或组件实例的直接操作。 在React开…

    2026年5月10日
    000
  • Redux Dispatch 不更新状态的排查与解决

    本文旨在帮助开发者诊断和解决 Redux 应用中 dispatch 函数调用后状态未更新的问题。通过分析常见的错误配置和代码实现,提供逐步排查方案和修正建议,确保 Redux 状态管理的正确性和可靠性。 在 Redux 应用开发中,dispatch 函数用于触发状态变更,如果 dispatch 调用…

    2026年5月10日
    100
  • React中正确处理Select元素OnChange事件

    在React应用中,正确监听select下拉菜单的值变化是常见的需求。本文将详细阐述,与原生HTML的onchange属性不同,React中应使用驼峰命名法的onChange属性来捕获此类事件。我们将通过示例代码演示如何结合React的状态管理,实现对select元素值的有效监听和响应,确保组件行为…

    2026年5月10日
    100
  • JavaScript中动态生成HTML链接:正确使用模板字面量嵌入URL

    本文深入探讨了在javascript中动态生成html链接时,如何正确地将变量(尤其是url)嵌入到`href`属性中。通过分析常见的错误,即混淆javascript的模板字面量与框架特有的模板语法,文章详细演示了使用es6模板字面量`${}`进行字符串插值的正确方法,确保动态链接能够被浏览器正确解…

    2026年5月10日
    000
  • HTML代码怎么实现错误边界_HTML代码错误边界处理方法与异常捕获策略

    答案:通过JavaScript模拟错误边界,结合try…catch、onerror事件、Promise.catch()及全局监控工具,可有效捕获并隔离HTML应用中的错误,防止功能失效。 HTML本身并没有直接提供像JavaScript那样的“错误边界”概念。HTML主要负责结构和内容,…

    2026年5月10日
    100
  • 在JavaScript中,如何实现数据的不可变性(Immutability)?

    使用const声明变量可防止重新赋值,但无法阻止对象内部修改,需结合扩展运算符、不可变数组方法和Object.freeze实现深层不可变,关键在于始终返回新对象而非修改原数据。 在JavaScript中,实现数据的不可变性意味着避免直接修改现有对象或数组,而是通过创建新对象来反映状态变化。虽然Jav…

    2026年5月10日
    200
  • 解决 React Native Android 应用启动时出现的伪启动页问题

    本文旨在解决 React Native Android 应用在启动时,先显示一个带有应用图标的黑色伪启动页,然后再显示自定义启动页的问题。通过修改 Android 项目的 `styles.xml` 文件,禁用应用的预览窗口,即可有效避免此问题,提升用户体验。 在开发 React Native 应用时…

    2026年5月10日
    100

发表回复

登录后才能评论
关注微信