源码分析之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关联起来。

相关推荐
鱼人1 天前
Web Components:未来的前端组件化标准?
前端
果汁华1 天前
Chrome DevTools MCP:让 AI 编码助手拥有浏览器调试超能力
前端·人工智能·chrome devtools
二月龙1 天前
移动端适配必杀技:Viewport与响应式布局全解
前端
大萝卜呼呼1 天前
Next.js第十七课 - 部署
前端·typescript·next.js
只会写Bug1 天前
后台管理项目中关于新增、编辑弹框使用的另一种展示形式
前端·vue.js
lion101 天前
简单Canvas指纹示例
javascript
weixin199701080161 天前
《废旧物资商品详情页前端性能优化实战》
前端·性能优化
用户52709648744901 天前
Vite 开发代理里的 `ws` 是什么,什么时候该开
前端
冰水不凉1 天前
robot_localization实现imu和odom融合
前端·slam
M ? A1 天前
Vue v-bind 转 React:VuReact 怎么处理?
前端·javascript·vue.js·经验分享·react.js·面试·vureact