Vue3 + TS 中使用 Provide/Inject 需要考虑的三大问题

Provide/Inject的作用

在组件通信的场景中,必然会遇到跨层级组件间传值的问题,尤其是爷------孙组件,甚至是更深层级的组件。比如下图中,App.vue 文件将属性 name 一层层地往下传给组件 form-item.vue:

对于这种类型的组件传值是有解决方法的,那就是使用 props 接受属性后,再不断的往下传,但是这种方式会非常麻烦,写很多不必要的代码;第二种方式是把要传递的属性放到全局状态管理当中,所有组件都能共享,但并不是适用所有的场景,有些单独的属性我们并不想放到全局状态管理器里面。

这个时候,Provide/Inject 的出现就是很好的解决深层组件间传值的问题,一句话概括 Provide/Inject 的作用:实现 Prop 逐级透传。

Provide/Inject 的简单使用:

js 复制代码
// App.vue
import { provide } from 'vue';
provide('name', 'zhangsan');

// form-item.vue
import { inject } from 'vue';
const name = inject('name');

在 Vue3 + TS 的项目中使用 Provide/Inject 其实是很简单的,但是如果我们想要用好这个 API,其实需要考虑到很多东西,尤其是在复杂的项目里面,那么这篇文章将给大家分享 使用 Provide/Inject 需要解决的三大问题。

问题一:命名冲突

在项目业务逻辑十分复杂,多人协作开发的情况下,很容易出现 provide 提供的 key 值发生冲突的问题,比如下面的场景:

title.vue 组件无法得知 name 属性是由哪个组件提供的,当然解决方法是非常多的,直接改个名称就行了,最简单的方式,但是复杂项目就不好处理了,而且可能其他同事在某个组件中 provide 的名称与我们的一样,还得麻烦自己去查找名称是否有冲突。

倒不如我们在命名的时候,取个唯一的名称,在 JS 里面 Symbol 类型的数据便可以帮我们解决这个问题。为了便于项目的规范,新建 inject-keys.ts 文件专门存放注入的名称:

js 复制代码
// src/inject-keys.ts
export const HomeNameKey = Symbol('name');
export const OtherNameKey = Symbol('name');

使用如下:

js 复制代码
// Home.vue
import { provide } from 'vue'; 
import { HomeNameKey } from '@/constants/inject-keys'; 
provide(HomeNameKey, 'hahaha');

// title.vue
import { inject } from 'vue';
import { HomeNameKey } from '@/constants/inject-keys';
const name = inject(HomeNameKey);

问题二:类型提示

对于 inject(HomeNameKey) 会得到一个 name 变量,这个时候如果我们想要知道 name 的类型是啥,就需要找到 provide(HomeNameKey, 'hahaha'),这样其实会比较麻烦,我们既然都使用 TS 进行项目的开发了,为什么不指定其类型呢?在组件中使用时自动获取数据的类型提示。

Vue3 中提供了 InjectionKey 用于定义注入变量的类型:

ts 复制代码
import { InjectionKey } from 'vue';

export type infoVO = {
    name: string;
    age: number;
}

export const InfoKey: InjectionKey<infoVO> = Symbol('info');

在组件中使用:

js 复制代码
// Home.vue
import { provide } from 'vue'; 
import { InfoKey } from '@/constants/inject-keys'; 

provide(InfoKey, 'zhangsan');  // 错误
provide(InfoKey, {name: 'hahah', age: 18});  // 正确

问题三:严格注入

在使用 inject 时会遇到一个问题,那就是如果注入的名称 InfoKey 在其祖先组件中并没有提供,那么 inject(InfoKey) 是会出现问题的,其实可以使用默认值来解决祖先组件未提供的情况,inject 这个 API 是可以接受三个参数的:

  • 第一个参数是注入的 key

  • 第二个参数是可选的,即在没有匹配到 key 时使用的默认值。

  • 第二个参数也可以是一个工厂函数,用来返回某些创建起来比较复杂的值。在这种情况下,你必须将 true 作为第三个参数传入,表明这个函数将作为工厂函数使用,而非值本身。

js 复制代码
// 注入一个值,若为空则使用提供的默认值 
const bar = inject('path', '/default-path') 
// 注入一个值,若为空则使用提供的函数类型的默认值 
const fn = inject('function', () => {}) 
// 注入一个值,若为空则使用提供的工厂函数 
const baz = inject('factory', () => new ExpensiveObject(), true)

但是有些情况下要求祖先链上必须提供需要的内容,尤其是在一些通用型组件的开发中,这个时候应该抛出错误而不是警告,因此需要解决这个问题。封装一个工具函数:

js 复制代码
export const injectStrict = <T>(key: InjectionKey<T>, defaultValue?: T | (() => T), treatDefaultAsFactory?: false): T => {
  const result = inject(key, defaultValue, treatDefaultAsFactory); 
  if (!result) { 
    throw new Error('xxxxxxxxxxxxx'); 
  } 
  return result;
}

使用:

js 复制代码
import { inject } from 'vue';
import { HomeNameKey } from '@/constants/inject-keys';
const name = injectStrict(HomeNameKey);
相关推荐
云水一下41 分钟前
Vue.js从零到精通系列(三):组件化基础——Props、Emits、插槽与生命周期
前端·javascript·vue.js
SEO_juper1 小时前
新独立站冷启动收录全攻略:配置、推送、抓取配额优化完整手册
前端·谷歌·seo·跨境电商·外贸·geo·独立站
TinssonTai1 小时前
这个 VS Code 插件让我的 AI Coding 又快又稳 - 旧瓶装新酒
前端·人工智能·程序员
体验家1 小时前
体验家 XMPlus 网页端问卷 SDK 技术解析:用几行 JavaScript 实现精准场景触发与防打扰机制
开发语言·前端·javascript
Maimai108081 小时前
Web3 前端交易系统如何落地:从下单 UI 到 Operation 编码、签名与实时状态更新
前端·react.js·ui·架构·前端框架·web3
kidding7232 小时前
高效备忘清单工具类小程序
前端·计算机网络·微信小程序·小程序
IMPYLH2 小时前
HTML 的 <abbr> 元素
前端·算法·html
李白的天不白2 小时前
Tree-Shaking
前端
Csvn2 小时前
TypeScript:你以为安全的 `JSON.parse` 其实是颗雷 — 运行时类型安全实战
前端·javascript
橘子星3 小时前
深入理解线性数据结构:栈、队列与链表
前端·javascript