您好,登錄后才能下訂單哦!
這篇文章將為大家詳細(xì)講解有關(guān)Vue 3.0中怎么實(shí)現(xiàn)依賴注入,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對(duì)相關(guān)知識(shí)有一定的了解。
使用過 Angular 的小伙伴對(duì) 依賴注入 應(yīng)該不會(huì)陌生,依賴注入簡(jiǎn)稱為 DI(Dependency Injection)。組件之間的依賴關(guān)系由容器在運(yùn)行期決定,形象的說,即由容器動(dòng)態(tài)的將某個(gè)依賴關(guān)系注入到組件之中。依賴注入的目的并非為軟件系統(tǒng)帶來更多功能,而是為了提升組件重用的頻率,并為系統(tǒng)搭建一個(gè)靈活、可擴(kuò)展的平臺(tái)。
在 Vue 3.0 中,為我們提供了簡(jiǎn)單的依賴注入功能 —— provide/inject。它們解決了以下問題:有一些深度嵌套的組件,而深層的子組件只需要父組件的部分內(nèi)容。在這種情況下,如果仍然將 prop 沿著組件鏈逐級(jí)傳遞下去,這樣使用起來會(huì)很麻煩。
為了解決上述問題,Vue 提供了 provide 和 inject。使用 provide/inject 之后,無論組件層次結(jié)構(gòu)有多深,父組件都可以作為其所有子組件的依賴提供者。
(圖片來源 —— https://v3.cn.vuejs.org/guide/component-provide-inject.html)
由上圖可知,在父組件上通過 provide 提供數(shù)據(jù),子組件通過 inject 注入數(shù)據(jù)。介紹完 provide/inject 的作用之后,我們來看一下具體的示例。
一、Provide/Inject 使用示例
在使用 provide/inject 特性時(shí),你可以通過 provide 和 inject 選項(xiàng)的方式來使用它。這種方式官方文檔已經(jīng)有介紹了,這里阿寶哥將介紹另一種使用方式,即在組合式 API 的 setup 組件選項(xiàng)中,通過 provide 和 inject 函數(shù)的方式來實(shí)現(xiàn)依賴注入。
<div id="app"></div> <script> const { createApp, h, provide, inject } = Vue const app = createApp({ render: () => h(Provider) }) const Provider = { setup() { provide('name', '阿寶哥') return () => h(Middle) } } const Middle = { render: () => h(Consumer) } const Consumer = { setup() { const name = inject('name') return () => `大家好,我是${name}!` } } app.mount('#app') </script>
在以上示例中,在 Provider 組件內(nèi)通過 provide 函數(shù)配置了數(shù)據(jù),而在 Consumer 組件中通過 inject 函數(shù)獲取 Provider 組件中已配置的數(shù)據(jù)。需要注意的是,示例中的 Consumer 組件是作為 Provider 組件的孫組件。因此,通過使用 provide/inject 提供的依賴注入功能,我們實(shí)現(xiàn)了數(shù)據(jù)的跨層級(jí)傳遞。
介紹完 provide 和 inject 函數(shù)的基本使用之后,接下來阿寶哥將帶大家一起來揭開它們背后的秘密。
二、Provide 函數(shù)
在分析 provide 函數(shù)之前,我們先來回顧一下它的用法:
const Provider = { setup() { provide('name', '阿寶哥') return () => h(Middle) } }
該函數(shù)被定義在 runtime-core/src/apiInject.ts 文件中:
// packages/runtime-core/src/apiInject.ts export interface InjectionKey<T> extends Symbol {} export function provide<T>(key: InjectionKey<T> | string | number, value: T) { if (!currentInstance) { if (__DEV__) { warn(`provide() can only be used inside setup().`) } } else { let provides = currentInstance.provides // 默認(rèn)情況下,組件實(shí)例會(huì)繼承于它父組件實(shí)例的 provides 對(duì)象,當(dāng)它需要為本身提供 // 值是,它將使用父組件的 provides 對(duì)象作為原型對(duì)象來創(chuàng)建屬于它自己的 provides // 對(duì)象。這樣的話,`inject` 函數(shù)就可以簡(jiǎn)單地從直接父對(duì)象中查找需注入的值,并讓原型鏈 // 來完成這個(gè)工作。 const parentProvides = currentInstance.parent && currentInstance.parent.provides if (parentProvides === provides) { provides = currentInstance.provides = Object.create(parentProvides) } // TS 不允許使用 symbol 作為索引類型 provides[key as string] = value } }
通過觀察以上的代碼,我們可以得出以下結(jié)論:
provide 函數(shù)只能使用在組合式 API 的 setup 函數(shù)中。
組件實(shí)例上會(huì)有一個(gè) provides 屬性,通過 provide 配置的數(shù)據(jù),最終會(huì)被保存到組件的 provides 屬性中。
provide 函數(shù)支持 3 種類型作為 key,即 InjectionKey
在以上代碼中,我們見到了 currentInstance 對(duì)象,那么這個(gè)對(duì)象內(nèi)部的結(jié)構(gòu)是怎樣的呢?為了一探究竟,我們?cè)?provide 函數(shù)內(nèi)部加個(gè)斷點(diǎn):
由上圖可知, currentInstance 是一個(gè)含有多種屬性的普通對(duì)象。其中 bc(BEFORE_CREATE)、bm(BEFORE_MOUNT) 和 bu(BEFORE_UPDATE) 是與生命周期相關(guān)的鉤子。那么問題又來了,currentInstance 對(duì)象是怎么創(chuàng)建的?看過之前 Vue 3.0 進(jìn)階系列文章的小伙伴,可能對(duì) createComponentInstance 函數(shù)有點(diǎn)印象,該函數(shù)的作用就是創(chuàng)建組件實(shí)例,具體代碼如下所示:
// packages/runtime-core/src/component.ts export function createComponentInstance( vnode: VNode, parent: ComponentInternalInstance | null, suspense: SuspenseBoundary | null ) { const type = vnode.type as ConcreteComponent // inherit parent app context - or - if root, adopt from root vnode const appContext = (parent ? parent.appContext : vnode.appContext) || emptyAppContext const instance: ComponentInternalInstance = { uid: uid++, vnode, type, parent, appContext, root: null!, // to be immediately set subTree: null!, // will be set synchronously right after creation update: null!, // will be set synchronously right after creation render: null, effects: null, provides: parent ? parent.provides : Object.create(appContext.provides), bc, bm, bu // 省略大部分屬性 } // 省略部分代碼 instance.root = parent ? parent.root : instance instance.emit = emit.bind(null, instance) return instance }
需要注意的是,如果當(dāng)前組件 parent 屬性的值不為 null 時(shí),則將使用 parent.provides 的值作為組件實(shí)例 provides 屬性的屬性值。介紹完 provide 和 createComponentInstance 函數(shù),為了讓大家能夠更好地理解前面的示例,阿寶哥用一張圖來總結(jié)一下示例中組件之間的關(guān)系:
對(duì)于根組件來說,它的 parent 屬性值為 null。好的,provide 函數(shù)就先介紹到這里,下面我們來開始介紹 inject 函數(shù)。
三、Inject 函數(shù)
同樣,在分析 inject 函數(shù)之前,我們也先來回顧一下它的用法:
const Consumer = { setup() { const name = inject('name') return () => `大家好,我是${name}!` } }
inject 函數(shù)與 provide 函數(shù)是互相配合的,它們都被定義在 runtime-core/src/apiInject.ts 文件中:
// packages/runtime-core/src/apiInject.ts export function inject( key: InjectionKey<any> | string, defaultValue?: unknown, treatDefaultAsFactory = false ) { const instance = currentInstance || currentRenderingInstance // 獲取當(dāng)前實(shí)例 if (instance) { // to support `app.use` plugins, // fallback to appContext's `provides` if the intance 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) { // TS doesn't allow symbol as index type return provides[key as string] } else if (arguments.length > 1) { return treatDefaultAsFactory && isFunction(defaultValue) ? defaultValue() : defaultValue } else if (__DEV__) { warn(`injection "${String(key)}" not found.`) } } else if (__DEV__) { warn(`inject() can only be used inside setup() or functional components.`) } }
在 inject 函數(shù)中,我們可以清楚地看到如果當(dāng)前實(shí)例的 parent 屬性為 null 時(shí),則會(huì)從 appContext 上下文中獲取 provides 對(duì)象,否則將從當(dāng)前實(shí)例的父組件實(shí)例中獲取 provides 對(duì)象。
對(duì)于我們的示例來說,在獲取到 provides 對(duì)象后,就會(huì)判斷 name 屬性是否存在于當(dāng)前的 provides 對(duì)象中,此時(shí)該對(duì)象是 { name: "阿寶哥"},所以會(huì)直接返回 "阿寶哥"。
此外,通過觀察 inject 函數(shù),我們還可以得出以下結(jié)論:
inject 函數(shù)的第二個(gè)參數(shù)是一個(gè)可選參數(shù) —— defaultValue?: unknown,用于設(shè)置默認(rèn)值或默認(rèn)值工廠。即在 provides 對(duì)象上找不到 key 對(duì)應(yīng)值的時(shí)候,可以使用默認(rèn)值或默認(rèn)值工廠的返回值來代替。
inject 函數(shù)只能在 setup() 或函數(shù)組件中使用。
四、App 對(duì)象中的 provide API
在創(chuàng)建完 Vue 3 應(yīng)用對(duì)象之后,我們可以使用該對(duì)象提供的 provide 方法。該方法設(shè)置一個(gè)可以被注入到應(yīng)用范圍內(nèi)所有組件中的值。之后,組件就可以使用 inject 來接收 provide 的值。
import { createApp } from 'vue' const app = createApp({ inject: ['name'], template: ` <div> {{ name }} </div> ` }) app.provide('name', '阿寶哥')
需要注意的是,app.provide 方法不應(yīng)該與 provide 組件選項(xiàng)或組合式 API 中的 provide 方法混淆。雖然它們也是相同的 provide/inject 機(jī)制的一部分,但是是用來配置組件 provide 的值而不是應(yīng)用 provide 的值。
介紹完 app.provide 方法之后,我們來了解一下它的實(shí)現(xiàn)??催^ ”Vue 3.0 進(jìn)階“ 系列教程的小伙伴,對(duì) app 對(duì)象應(yīng)該不會(huì)陌生。因?yàn)樵谇懊娴奈恼轮?,阿寶哥已?jīng)介紹過 component、directive 和 mount 等方法。接下來,我們來看一下 provide 方法的具體實(shí)現(xiàn):
// packages/runtime-core/src/apiCreateApp.ts export function createAppAPI<HostElement>( render: RootRenderFunction, hydrate?: RootHydrateFunction ): CreateAppFunction<HostElement> { return function createApp(rootComponent, rootProps = null) { const context = createAppContext() // 省略大部分內(nèi)容 const app: App = (context.app = { _uid: uid++, _context: context, provide(key, value) { // TypeScript doesn't allow symbols as index type // https://github.com/Microsoft/TypeScript/issues/24587 context.provides[key as string] = value return app } }) return app } }
由以上代碼可知,在 provide 方法內(nèi)部會(huì)把 key 和 value 以鍵值對(duì)的形式保存在應(yīng)用上下文 context 對(duì)象的 provides 屬性中。
// packages/runtime-core/src/apiCreateApp.ts export function createAppContext(): AppContext { return { app: null as any, config: { ... }, mixins: [], components: {}, directives: {}, provides: Object.create(null) } }
五、阿寶哥有話說
5.1 在嵌套的 providers 場(chǎng)景下,存在同名的 key 會(huì)怎么樣?
<div id="app"></div> <script> const { createApp, h, provide, inject } = Vue const app = createApp({ render: () => h(ProviderOne) }) const ProviderOne = { setup() { provide('foo', 'foo') provide('bar', 'bar') return () => h(ProviderTwo) } } const ProviderTwo = { setup() { provide('foo', 'fooOverride') provide('baz', 'baz') return () => h(Consumer) } } const Consumer = { setup() { const foo = inject('foo') const bar = inject('bar') const baz = inject('baz') return () => [foo, bar, baz].join(',') } } app.mount('#app') </script>
在以上代碼中,ProviderOne 和 ProviderTwo 組件中使用同樣的 foo 屬性名配置了 Provider。然后,我們?cè)诘讓拥? Consumer 組件中使用 inject API 分別注入了 ProviderOne 和 ProviderTwo 中配置的值。接下來,我們先來看一下結(jié)果:
fooOverride,bar,baz
由以上結(jié)果可知,在嵌套 providers 的場(chǎng)景中,會(huì)就近從父組件實(shí)例中獲取對(duì)應(yīng)的值,找不到的話,會(huì)往上一層層進(jìn)行查找。
5.2 通過 inject 獲取響應(yīng)式的值,能否正常工作?
在某些場(chǎng)景下,我們希望往深層的子組件傳遞通過響應(yīng)式 API 創(chuàng)建的響應(yīng)式的值,那么通過 inject 函數(shù)獲取的響應(yīng)式的值可以正常工作么?要弄清楚這個(gè)問題,我們來看一個(gè)具體的示例:
<div id="app"></div> <script> const { createApp, h, provide, inject, ref, onMounted } = Vue const nameRef = ref("阿寶哥"); const app = createApp({ setup() { provide("name", nameRef); return () => h(Middle) } }) const Middle = { render: () => h(Consumer) } const Consumer = { setup() { const name = inject('name') onMounted(() => { setTimeout(() => nameRef.value = "kakuqo", 2000); }) return () => `大家好,我是${name.value}!` } } app.mount('#app') </script>
在以上代碼中,我們通過 ref API 創(chuàng)建了一個(gè) nameRef 對(duì)象,然后在根組件中通過 provide 函數(shù)配置相應(yīng)的 Provider。而在 Consumer 組件的 setup 方法內(nèi),我們通過 inject 函數(shù)注入了 nameRef 對(duì)象,并通過 name.value 訪問了該對(duì)象內(nèi)保存的值。
此外,在 setup 方法內(nèi)部,我們還使用了 onMounted 生命周期鉤子,在鉤子對(duì)應(yīng)的回調(diào)函數(shù)中,我們延遲 2S 修改 nameRef 對(duì)象的值。
以上示例成功運(yùn)行后,首先會(huì)先顯示 大家好,我是阿寶哥!,差不多 2S 后頁面會(huì)刷新為 大家好,我是kakuqo!。
5.3 是否支持 self-inject?
什么是 self-inject 呢?這里阿寶哥不做過多解釋,我們直接來看個(gè)具體的例子:
<div id="app"></div> <script> const { createApp, h, provide, inject } = Vue const app = createApp({ render: () => h(Provider) }) const Provider = { setup() { provide('name', '阿寶哥') const injectedName = inject('name') return () => h(injectedName) } } app.mount('#app') </script>
在以上代碼中,我們?cè)?Provider 組件的 setup 方法內(nèi)部先使用 provide 函數(shù)配置了相應(yīng)的 Provider,然后使用 inject 函數(shù)來獲取對(duì)應(yīng)的值。很明顯這個(gè)操作并沒有實(shí)際的意義,那么可以這樣使用么?答案是可以的,以上示例成功運(yùn)行之后,Provider 組件會(huì)被轉(zhuǎn)換為 注釋節(jié)點(diǎn)。
<div id="app" data-v-app=""><!----></div>
那么為什么會(huì)轉(zhuǎn)為注釋節(jié)點(diǎn)呢?因?yàn)?injectedName 的值為 undefined,在通過 h 函數(shù)創(chuàng)建 VNode 對(duì)象的時(shí)候,會(huì)繼續(xù)調(diào)用 createVNode 函數(shù),在該函數(shù)內(nèi)部如果發(fā)現(xiàn)是 type 類型為 falsy 值時(shí),會(huì)把 VNode 對(duì)象的類型統(tǒng)一轉(zhuǎn)換為 Comment 類型。
// packages/runtime-core/src/vnode.ts function _createVNode( type: VNodeTypes | ClassComponent | typeof NULL_DYNAMIC_COMPONENT, props: (Data & VNodeProps) | null = null, children: unknown = null, patchFlag: number = 0, dynamicProps: string[] | null = null, isBlockNode = false ): VNode { if (!type || type === NULL_DYNAMIC_COMPONENT) { if (__DEV__ && !type) { warn(`Invalid vnode type when creating vnode: ${type}.`) } type = Comment } // 省略大部分代碼 }
關(guān)于Vue 3.0中怎么實(shí)現(xiàn)依賴注入就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。