Vue.js 3.x版本中,不同层级组件间数据传递方法有哪些?

2026-05-06 04:401阅读0评论SEO教程
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计1125个文字,预计阅读时间需要5分钟。

Vue.js 3.x版本中,不同层级组件间数据传递方法有哪些?

在Vue.js 3.x中,`provide/inject`的实现原理是通过响应式系统进行跨层级数据传递。`provide`和`inject`允许父组件向任意深度的子组件提供数据,而无需逐层手动传递。

`provide/inject`如何实现跨层级传递数据:

1.父组件通过`provide`选项提供数据。

2.子组件通过`inject`选项接收数据。

基本用法:

在Vue.js中,跨层级组件传递数据时,可以使用`props`。如果层级较深,`props`传递数据会变得繁琐。这时,`provide/inject`就派上了用场。

示例:javascript// 父组件

// 子组件 {{ data }}

在上述示例中,父组件通过`provide`向子组件提供数据,子组件通过`inject`接收数据。这样,我们就可以在任意深度的组件之间传递数据,而无需逐层手动传递。

在 Vue.js 3.x 中,provide/inject 的实现原理是什么?provide/inject 是如何做到跨层级传递数据的? provide/inject 基本用法

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

注:上图来自 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。

现在,问题解决了:

注:上图来自 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 的代码也很简单,简单到你看了之后会来一句:

Vue.js 3.x版本中,不同层级组件间数据传递方法有哪些?

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 利用原型链来实现跨层级组件的数据传递。

本文共计1125个文字,预计阅读时间需要5分钟。

Vue.js 3.x版本中,不同层级组件间数据传递方法有哪些?

在Vue.js 3.x中,`provide/inject`的实现原理是通过响应式系统进行跨层级数据传递。`provide`和`inject`允许父组件向任意深度的子组件提供数据,而无需逐层手动传递。

`provide/inject`如何实现跨层级传递数据:

1.父组件通过`provide`选项提供数据。

2.子组件通过`inject`选项接收数据。

基本用法:

在Vue.js中,跨层级组件传递数据时,可以使用`props`。如果层级较深,`props`传递数据会变得繁琐。这时,`provide/inject`就派上了用场。

示例:javascript// 父组件

// 子组件 {{ data }}

在上述示例中,父组件通过`provide`向子组件提供数据,子组件通过`inject`接收数据。这样,我们就可以在任意深度的组件之间传递数据,而无需逐层手动传递。

在 Vue.js 3.x 中,provide/inject 的实现原理是什么?provide/inject 是如何做到跨层级传递数据的? provide/inject 基本用法

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

注:上图来自 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。

现在,问题解决了:

注:上图来自 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 的代码也很简单,简单到你看了之后会来一句:

Vue.js 3.x版本中,不同层级组件间数据传递方法有哪些?

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 利用原型链来实现跨层级组件的数据传递。