源码分析之React中的useImperativeHandle

概述

useImperativeHandle用子组件安全地向父组件提供访问方法或者变量的句柄。通常和React.forwardRef配合使用。

源码分析

useImperactiveHandle同其他hooks 一样,在挂载和更新阶段是对应不同的方法,分别为:mountImperativeHandleupdateImperativeHandle

源码解读

  • mountImperativeHandle方法
js 复制代码
function mountImperativeHandle(ref, create, deps) {
  const effectDeps =
    deps !== null && deps !== undefined ? deps.concat([ref]) : null;

  let fiberFlags = UpdateEffect | Layout;

  mountEffectImpl(
    fiberFlags,
    HookLayout,
    imperativeHandleEffect.bind(null, create, ref),
    effectDeps,
  );
}
  • updateImperativeHandle方法
js 复制代码
function updateImperativeHandle(ref, create, deps) {
  const effectDeps =
    deps !== null && deps !== undefined ? deps.concat([ref]) : null;

  updateEffectImpl(
    UpdateEffect,
    HookLayout,
    imperativeHandleEffect.bind(null, create, ref),
    effectDeps,
  );
}

可以明显看出,上面两个方法和effect的挂载更新几乎如出一辙,都是调用mountEffectImplupdateEffectImpl进行处理,而且useImperativeHandleuseLayoutEffect使用的tag标记是一样的,因为useImperativeHandle的本质给ref参数赋值,而useLayoutEffect这个hook是在DOM挂载后、DOM绘制前调用,tag一致可以保证useImperativeHandle的调用时机也是这个时机。

另一方面,useRef这个hook也可以用于绑定DOM,这也遵守了相同规则。

useLayoutEffect不同的是,useImperativeHandle会自动的将ref也作为依赖的一部分,并且通过bind方法对create回调方法进行了封装。当依赖发生变化时,便会调用imperativeHandleEffect方法,在其内对ref进行赋值。

其源码如下:

  • imperativeHandleEffect方法
js 复制代码
function imperativeHandleEffect(create, ref) {
  // 判断ref是否是函数
  if (typeof ref === 'function') {
   // 若ref是函数
    const refCallback = ref;
    // 先执行create方法 
    const inst = create();
    // 然后将create方法的返回值作为ref的参数,执行ref函数
    const refCleanup = refCallback(inst);// 拿到ref函数的返回值
    // 返回一个函数,作为useImperative的清理函数
    return () => {
      // 判断,若ref的返回值是一个函数
      if (typeof refCleanup === 'function') {
       // 则在useImperativeHandle调用destroy时,执行对ref的清理
        refCleanup();
      } else {
        // 否则再次调用ref函数,传值为null
        refCallback(null);
      }
    };
  } else if (ref !== null && ref !== undefined) {
   // 若ref是一个对象,则其必然是一个{current:any}的对象,否则React会报错
    const refObject = ref;
    // 执行create方法,并将其返回值赋值给re.current
    const inst = create();
    refObject.current = inst;
    // 清理函数就是将ref,current置为null
    return () => {
      refObject.current = null;
    };
  }
}

总结

useImperativeHandle的设计沿用了useLayoutEffect,其主要作用就是搭配React.forwardRef向父组件注入调用子组件的方法或数据。

相关推荐
漂流瓶jz5 小时前
总结CSS组件化演进之路:命名规范/CSS Modules/CSS in JS/原子化CSS
前端·javascript·css
踩着两条虫5 小时前
「AI + 低代码」的可视化设计器
开发语言·前端·低代码·设计模式·架构
JoneBB5 小时前
ABAP Webservice连接
运维·开发语言·数据库·学习
Jagger_5 小时前
项目上线忙碌结束之后,为什么总想找点事做?
前端
GalenZhang8886 小时前
OpenClaw 配置多个飞书账号实战指南
前端·chrome·飞书·openclaw
即使再小的船也能远航6 小时前
【Python】安装
开发语言·python
Irissgwe6 小时前
类与对象(三)
开发语言·c++·类和对象·友元
steven~~~6 小时前
为什么mq报错
javascript
雪度娃娃6 小时前
转向现代C++——优先选用nullptr而不是0和NULL
开发语言·c++
萌新小码农‍7 小时前
python装饰器
开发语言·前端·python