• Vue.js 3.x 中跨层级组件如何传递数据?


    provide/inject 基本用法

    Vue.js 中,跨层级组件如果想要传递数据,我们可以直接使用 props 来将祖先组件的数据传递给子孙组件:

    prop-drilling.png

    注:上图来自 Vue.js 官网:Prop Drilling

    如上图所示,中间组件 <Footer> 可能根本不需要这部分 props,但为了 <DeepChiild> 能访问这些 props<Footer> 还是需要定义这些 props,并将其传递下去。

    有人说我们可以使用 $attrs/$listeners,但依然还要经过中间层级,而使用 Vuex 又过于麻烦,Event Bus 又很容易导致逻辑分散,出现问题后难以定位。

    那么,有没有其他方法可以实现直接从祖先组件传递数据给子孙组件呢?答案就是 provide/inject

    祖先组件:

    // Root.vue
    <script setup>
    import { provide } from 'vue'
    provide('msg' /* 注入的键名 */ , 'Vue.js' /* 值 */)
    </script>

    子孙组件:

    // DeepChild.vue
    <script setup>
    import { inject } from 'vue'
    const msg = inject('msg' /* 注入的键名 */, 'World' /* 默认值 */)
    </script>

    具体用法详见:Provide / Inject

    现在,问题解决了:

    prop-drilling2.png

    注:上图来自 Vue.js 官网:Prop Drilling

    provide 实现原理

    这么神奇的东西,究竟是如何实现的呢?

    export function provide<T>(key: InjectionKey<T> | string | number, value: T) {
    let provides = currentInstance.provides
    const parentProvides = currentInstance.parent && currentInstance.parent.provides
    if (parentProvides === provides) {
    provides = currentInstance.provides = Object.create(parentProvides)
    }
    provides[key as string] = value
    }

    在默认情况下,组件实例的 provides 继承自其父组件。但是当组件实例需要提供自己的值的时候,它使用父组件的 provides 对象作为原型,来创建自己的 provides 对象。这样一来,当使用 inject 时,我们就可以通过原型链来找到父组件提供的数据

    inject 实现原理

    inject 的代码也很简单,简单到你看了之后会来一句:

    就这?

    export function inject(
    key: InjectionKey<any> | string,
    defaultValue?: unknown,
    treatDefaultAsFactory = false
    ) {
    const instance = currentInstance || currentRenderingInstance
    if (instance) {
    // #2400
    // to support `app.use` plugins,
    // fallback to appContext's `provides` if the instance is at root
    const provides = instance.parent == null
    ? instance.vnode.appContext && instance.vnode.appContext.provides
    : instance.parent.provides
    if (provides && (key as string | symbol) in provides) {
    return provides[key as string]
    } else if (arguments.length > 1) {
    return treatDefaultAsFactory && isFunction(defaultValue)
    ? defaultValue.call(instance.proxy)
    : defaultValue
    }
    }
    }

    inject 的主要功能就两点:

    • 通过 in 操作获取父组件的数据,in 操作会遍历原型链,这就是上面 provide 的实现中,为什么组件要使用父组件的 provides 对象作为原型来创建自己 provides 对象的原因
    • 实现 inject 的默认值功能,inject 第二个参数为默认值

    一句话总结:provide/inject 利用原型链来实现跨层级组件的数据传递。

  • 相关阅读:
    springcloud9:openFeign
    leetcode刷题(133)——剑指 Offer 07. 重建二叉树
    【大道模式】状态模式 - State Pattern(审核状态流转)
    反射(获取成员变量,获取成员方法)
    (09_22)【有奖体验】轻点鼠标,让古籍数字化“重生_
    多表查询和连接查询
    计算机毕业设计Javam和vue的酒店管理系统2021(源码+系统+mysql数据库+lw文档)
    html中使用vue组件
    php-面向对象OOP
    UNIX网络模型
  • 原文地址:https://www.cnblogs.com/ashengtan/p/16221057.html