源码分析之React中useRef解析

流程分析

useRef架构图

useRef的流程架构图如下:




旧ref存在
新ref存在
更新:增删改
updateRef方法
旧fiber存在且旧ref不存在?(ref删除了)
首次渲染或者新旧ref不一样
首次渲染
mountRef方法
beginWork阶段
markRef检测

workInProgress.ref==null
workInProgress.flags打标记flag
commit阶段
commitMutationEffectsOnFiber
满足current!== null等条件才调用:safelyDetachRef解绑
current.ref!=null?
解绑操作
commitLayoutEffectOnFiber
commitAttachRef
finishedWork.ref!=null?
绑定赋值

相关方法解析

mountRef方法

在首次渲染挂载时,useRef调用的是mountRef

js 复制代码
function mountRef(initialValue) {
  // 创建hook,将hook绑定fiber的updateQueue上并返回
  const hook = mountWorkInProgressHook();
  // 包装ref值
  const ref = { current: initialValue };
  // 将值保存到hook.memoizedState上
  hook.memoizedState = ref;
  return ref;
}
updateRef
js 复制代码
function updateRef(initialValue) {
  // 获取fiber上的新hook,新hook来自于旧fiber上的旧hook 
  const hook = updateWorkInProgressHook();
  // 返回新hook的memoizedState
  return hook.memoizedState;
}

通过mountRefupdateRef可知,userRefhookupdateQueue上就是一个赋值/取值的操作。

markRef方法

useRef的标记发生在beginWork 阶段,会调用markRef方法,在组件更新时,检查ref是否发生变化,并给Fiber节点打上"需要更新ref"的标记。

js 复制代码
function markRef(current, workInProgress) {
  // 取出新fiber上的ref
  const ref = workInProgress.ref;
  // 若新fiber上没有ref
  if (ref === null) {
    // 旧fiber存在,且存在ref
    if (current !== null && current.ref !== null) {
     // 则说明ref被删除了,打上标记,后续需要清空旧ref
      workInProgress.flags |= Ref | RefStatic;
    }
  } else {
    // 判断ref的类型是否是函数或对象,若不是,则报错
    if (typeof ref !== 'function' && typeof ref !== 'object') {
      throw new Error(
        'Expected ref to be a function, an object returned by React.createRef(), or undefined/null.',
      );
    }
    // 组件首次渲染时,或者是新旧ref不同,则也需要打上标记
    if (current === null || current.ref !== ref) {
      workInProgress.flags |= Ref | RefStatic;
    }
  }
}

首次渲染或者新旧fiberref不等,以及ref被删除,在fiber.flags上的标记都是一样。

safelyDetachRef方法

commitMutationEffectsOnFiber中调用safelyDetachRef方法,若当前不是首次渲染且fiber.flagRef标记则需要清理ref,此外在组件卸载时也需要调用fiber.flag,释放对DOM的引用。

js 复制代码
function safelyDetachRef(
  current
) {
// current是指当前UI展示的fiber
  const ref = current.ref;
  const refCleanup = current.refCleanup;
// 若ref存在,且fiber.flags上标记了需要更新,则先进行ref的清理
  if (ref !== null) {
   // 先判断清理方法是否是函数
    if (typeof refCleanup === 'function') {
      try {
      // 执行清理函数
        refCleanup();
      } catch (error) {
        // 报错
      } finally {
        // 重置新fiber、旧fiber上的ref清理函数 
        current.refCleanup = null;
        const finishedWork = current.alternate;
        if (finishedWork != null) {
          finishedWork.refCleanup = null;
        }
      }
    } else if (typeof ref === 'function') {
    // 判断 ref是否是函数
      try {
      // 是函数,则再调用一遍
        ref(null);
      } catch (error) {
        // 报错
      }
    } else {
    // 若ref不是函数,则将ref.current重置
      ref.current = null;
    }
  }
}
绑定赋值

一般在commitLayoutEffectOnFiber中调用,此时已经可以拿到DOM了。

js 复制代码
function commitLayoutEffectLayoutOnFiber(
  finishedRoot,
  current,
  finishedWork,
  committedLanes,
) {
  const flags = finishedWork.flags;
  switch (finishedWork.tag) {
    case ClassComponent: {
      //...
      if (flags & Ref) {
        safelyAttachRef(finishedWork, finishedWork.return);
      }
    }
    case HostHoistable:
    case HostComponent: {
      if (flags & Ref) {
        safelyAttachRef(finishedWork, finishedWork.return);
      }
    }
  }
}

如下的safelyAttachRef本质上就是调用commitAttachRef方法

js 复制代码
function commitAttachRef(finishedWork) {
  const ref = finishedWork.ref;
  if (ref !== null) {
    let instanceToUse;
    switch (finishedWork.tag) {
      case HostHoistable:
      case HostSingleton:
      case HostComponent:
        instanceToUse = finishedWork.stateNode;
        break;
      case ViewTransitionComponent: {
       
        instanceToUse = finishedWork.stateNode;
        break;
      }
      default:
        instanceToUse = finishedWork.stateNode;
    }
    // 判断ref的类型
    if (typeof ref === 'function') {
      // 只有ref为函数且有返回值,返回值需要是函数,才可以在后续执行ref清理  
      finishedWork.refCleanup = ref(instanceToUse);
    } else {
      ref.current = instanceToUse;
    }
  }
}
注意事项

safelyDetachRefcommitAttachRef方法中操作的都是fiber.ref,它与hook.memoizedState实际上是同一个引用地址,如下:

js 复制代码
function FC()=>{
    const divRef=useRef();
    return <div ref={divRef}>Hello word</div>    
}

ref={divRef}就是将hook.memoziedStatefiber.ref关联起来。

相关推荐
七牛云行业应用7 分钟前
别每次重复配置了!CLAUDE.md + Hooks 让 Claude Code 开箱就记住你的规则
前端
超人气王14 分钟前
新手学前端 JavaScript 类型判断:一篇彻底搞懂 typeof、instanceof 和 Object.prototype.toString
前端·javascript
LucianaiB21 分钟前
耗时30天,DocPilot Qwen正式开源:一个免费无广的开源文档 AI 助手
前端·后端
xiaoshuaishuai839 分钟前
C# AvaloniaUI 资源找不到报错
java·服务器·前端·windows·c#
丷丩1 小时前
MapLibre GL JS第35课:显示带地形高程(三维地形)的卫星影像
javascript·gis·map·mapbox·maplibre gl js
How_doyou_do1 小时前
26字节工程营-前端-自我总结
前端
三乐2281 小时前
node不认识类型?多半是没用上这几段代码
javascript
十有八七1 小时前
🧩 组件库死亡倒计时?—— AI 编码冲击下的前端基础设施重构
前端·人工智能
风止何安啊1 小时前
我一个前端仔,居然用 Python 搞起了 AI?从零到一,撸了个 AI 聊天框小 demo
前端·人工智能·后端
GISer_Jing1 小时前
Claude Code插件系统全解析
前端·人工智能·ai·架构