React19源码系列之Hooks(useRef)

在上一篇文章中,介绍了useId的使用及其原理,这次来看一下useRef的使用和它的原理。

useRef的使用

https://zh-hans.react.dev/reference/react/useRef

1、 引用 DOM 元素

useRef 还可以用于引用 DOM 元素,以便在组件中直接操作这些元素。例如,获取输入框的值

js 复制代码
import React, { useRef } from 'react';

function InputExample() {
    const inputRef = useRef(null);

    const handleClick = () => {
        console.log('输入框的值:', inputRef.current.value);
    };

    return (
        <div>
            <input type="text" ref={inputRef} />
            <button onClick={handleClick}>获取输入框的值</button>
        </div>
    );
}

export default InputExample;

2、 存储可变变量

在组件中,有时候需要存储一些变量,这些变量的变化不希望触发组件的重新渲染。例如,在一个定时器中记录当前的计数:

js 复制代码
import React, { useRef, useEffect } from 'react';

function Timer() {
    const countRef = useRef(0);

    useEffect(() => {
        const intervalId = setInterval(() => {
            countRef.current++;
            console.log('Count:', countRef.current);
        }, 1000);

        return () => clearInterval(intervalId);
    }, []);

    return <div>定时器正在运行...</div>;
}

export default Timer;

3、ref传给子组件,父组件中调用子组件方法

js 复制代码
import React, { useRef, useImperativeHandle } from 'react';

const ChildFunctionalComponent = (props) => {
   const { ref } = props
    const internalRef = useRef(null);

    useImperativeHandle(ref, () => ({
        focus: () => {
            internalRef.current.focus();
        }
    }));

    return <input type="text" ref={internalRef} />;
};

function ParentFunctionalComponent() {
    const childRef = useRef(null);

    const handleClick = () => {
        // 调用子组件暴露的方法
        childRef.current.focus();
    };

    return (
        <div>
            <ChildFunctionalComponent ref={childRef} />
            <button onClick={handleClick}>聚焦子组件输入框</button>
        </div>
    );
}

export default ParentFunctionalComponent;

useRef源码

useRef 是 React 提供的一个内置 Hook,它返回一个可变的 ref 对象,该对象具有一个 current 属性,初始值为传入的 initialValue。这个 ref 对象在组件的整个生命周期内保持不变,常用于在组件中存储不需要触发重新渲染的变量,或者引用 DOM 元素。

js 复制代码
function useRef<T>(initialValue: T): {current: T} {
  // resolveDispatcher 是一个内部函数,其作用是获取当前的 React 调度器。调度器负责管理 React 的渲染和更新过程,不同的环境(如开发环境、生产环境、服务器端渲染等)可能使用不同的调度器。
  const dispatcher = resolveDispatcher();
  // 调用获取到的调度器的 useRef 方法,并将 initialValue 作为参数传递给它。
  return dispatcher.useRef(initialValue);
}

mountRef 在函数组件初始化时遇到useRef触发

mountRef 函数是 React 内部用于在组件挂载阶段创建和初始化 ref 对象的函数。在 React 中,ref 是一种用于访问 DOM 节点或组件实例的方式,它提供了一种在函数式组件中存储和访问可变值的方法。

mountRef 函数的主要作用是创建一个 ref 对象,并将其存储在当前正在处理的钩子(hook)的 memoizedState 中,以便在组件的整个生命周期内保持该 ref 对象的引用。

函数数参数含义:

  • initialValue:类型为 T,是 ref 对象的初始值。这个值会被赋值给 ref 对象的 current 属性。
js 复制代码
function mountRef<T>(initialValue: T): {current: T} {
  // mountWorkInProgressHook 是 React 内部的一个函数,用于在当前正在处理的 Fiber 节点上创建一个新的钩子实例。每个钩子实例都有自己的状态和相关信息,存储在 hook 对象中。
  const hook = mountWorkInProgressHook();
  const ref = {current: initialValue};
  // 存储 ref 对象到钩子的 memoizedState 中
  hook.memoizedState = ref;
  // 返回 ref 对象
  return ref;
}

updateRef在函数组件更新时遇到useRef触发

updateRef 函数是 React 内部用于在组件更新阶段处理 ref 的函数。在 React 中,ref 提供了一种访问 DOM 节点或组件实例的方式,并且在组件的整个生命周期内保持引用的稳定性。
updateRef 函数的主要作用是在组件更新时,从当前正在处理的钩子(hook)的 memoizedState 中获取之前存储的 ref 对象并返回,确保在组件更新过程中使用的是同一个 ref 对象。

函数参数含义:

  • initialValue:类型为 T,理论上是 ref 对象的初始值,但在 updateRef 函数中,该参数并没有实际被使用,因为在更新阶段,ref 对象已经在挂载阶段创建好了,这里只是获取之前创建的 ref 对象。
js 复制代码
function updateRef<T>(initialValue: T): {current: T} {
  // updateWorkInProgressHook 是 React 内部的一个函数,用于在当前正在处理的 Fiber 节点上获取当前的钩子实例。
  const hook = updateWorkInProgressHook();
  // 返回存储在钩子中的 ref 对象
  return hook.memoizedState;
}

mountWorkInProgressHook函数和updateWorkInProgressHook函数的具体实现,可看上一篇useId的分析

相关推荐
老毛肚8 小时前
jeecg-boot-base-core 02 day
javascript·python
烬羽13 小时前
后端返回的 JSON 字符串,浏览器怎么"看懂"的?——Ajax 全链路拆解
javascript
半个落月14 小时前
一个新手用 Bun + Axios 调通 DeepSeek API 的实践记录
javascript
不好听61314 小时前
深入理解链表:线性数据结构的另一面
javascript·数据结构
林希_Rachel_傻希希14 小时前
学React治好了我的焦虑症,1小时速通React 前20分钟。
前端·javascript·面试
小林ixn14 小时前
从 Ajax 到异步编程:JSON 序列化、Event Loop 与 XHR 请求完全解析
javascript
丷丩16 小时前
MapLibre GL JS第47课:添加动画图标
javascript·gis·动画·mapbox·maplibre
快乐的哈士奇16 小时前
【Next.js实战①】Gmail API 按柜号检索邮件:OAuth 双 Cookie 与搜索 Fallback
开发语言·javascript·ecmascript
云水一下16 小时前
Vue.js从零到精通系列(五):全局状态管理——Pinia 核心与实践
前端·javascript·vue.js
kmblack117 小时前
javascript计算年龄
开发语言·javascript·ecmascript