Vue3 + TS 中使用 Provide/Inject 需要考虑的三大问题
ztj100 2024-11-12 14:22 16 浏览 0 评论
Provide/Inject的作用
在组件通信的场景中,必然会遇到跨层级组件间传值的问题,尤其是爷——孙组件,甚至是更深层级的组件。比如下图中,App.vue 文件将属性 name 一层层地往下传给组件 form-item.vue:
对于这种类型的组件传值是有解决方法的,那就是使用 props 接受属性后,再不断的往下传,但是这种方式会非常麻烦,写很多不必要的代码;第二种方式是把要传递的属性放到全局状态管理当中,所有组件都能共享,但并不是适用所有的场景,有些单独的属性我们并不想放到全局状态管理器里面。
这个时候,Provide/Inject 的出现就是很好的解决深层组件间传值的问题,一句话概括 Provide/Inject 的作用:实现 Prop 逐级透传。
Provide/Inject 的简单使用:
// 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 文件专门存放注入的名称:
// src/inject-keys.ts
export const HomeNameKey = Symbol('name');
export const OtherNameKey = Symbol('name');
使用如下:
// 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 用于定义注入变量的类型:
import { InjectionKey } from 'vue';
export type infoVO = {
name: string;
age: number;
}
export const InfoKey: InjectionKey<infoVO> = Symbol('info');
在组件中使用:
// 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 作为第三个参数传入,表明这个函数将作为工厂函数使用,而非值本身。
// 注入一个值,若为空则使用提供的默认值
const bar = inject('path', '/default-path')
// 注入一个值,若为空则使用提供的函数类型的默认值
const fn = inject('function', () => {})
// 注入一个值,若为空则使用提供的工厂函数
const baz = inject('factory', () => new ExpensiveObject(), true)
但是有些情况下要求祖先链上必须提供需要的内容,尤其是在一些通用型组件的开发中,这个时候应该抛出错误而不是警告,因此需要解决这个问题。封装一个工具函数:
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;
}
使用:
import { inject } from 'vue';
import { HomeNameKey } from '@/constants/inject-keys';
const name = injectStrict(HomeNameKey);
相关推荐
- Vue3非兼容变更——函数式组件(vue 兼容)
-
在Vue2.X中,函数式组件有两个主要应用场景:作为性能优化,因为它们的初始化速度比有状态组件快得多;返回多个根节点。然而在Vue3.X中,有状态组件的性能已经提高到可以忽略不计的程度。此外,有状态组...
- 利用vue.js进行组件化开发,一学就会(一)
-
组件原理/组成组件(Component)扩展HTML元素,封装可重用的代码,核心目标是为了可重用性高,减少重复性的开发。组件预先定义好行为的ViewModel类。代码按照template\styl...
- Vue3 新趋势:10 个最强 X 操作!(vue.3)
-
Vue3为前端开发带来了诸多革新,它不仅提升了性能,还提供了...
- 总结 Vue3 组件管理 12 种高级写法,灵活使用才能提高效率
-
SFC单文件组件顾名思义,就是一个.vue文件只写一个组件...
- 前端流行框架Vue3教程:17. _组件数据传递
-
_组件数据传递我们之前讲解过了组件之间的数据传递,...
- 前端流行框架Vue3教程:14. 组件传递Props效验
-
组件传递Props效验Vue组件可以更细致地声明对传入的props的校验要求...
- 前端流行框架Vue3教程:25. 组件保持存活
-
25.组件保持存活当使用...
- 5 个被低估的 Vue3 实战技巧,让你的项目性能提升 300%?
-
前端圈最近都在卷性能优化和工程化,你还在用老一套的Vue3开发方法?作为摸爬滚打多年的老前端,今天就把私藏的几个Vue3实战技巧分享出来,帮你在开发效率、代码质量和项目性能上实现弯道超车!一、...
- 绝望!Vue3 组件频繁崩溃?7 个硬核技巧让性能暴涨 400%!
-
前端的兄弟姐妹们五一假期快乐,谁还没在Vue3项目上栽过跟头?满心欢喜写好的组件,一到实际场景就频频崩溃,页面加载慢得像蜗牛,操作卡顿到让人想砸电脑。用户疯狂吐槽,领导脸色难看,自己改代码改到怀疑...
- 前端流行框架Vue3教程:15. 组件事件
-
组件事件在组件的模板表达式中,可以直接使用...
- Vue3,看这篇就够了(vue3 从入门到实战)
-
一、前言最近很多技术网站,讨论的最多的无非就是Vue3了,大多数都是CompositionAPI和基于Proxy的原理分析。但是今天想着跟大家聊聊,Vue3对于一个低代码平台的前端更深层次意味着什么...
- 前端流行框架Vue3教程:24.动态组件
-
24.动态组件有些场景会需要在两个组件间来回切换,比如Tab界面...
- 前端流行框架Vue3教程:12. 组件的注册方式
-
组件的注册方式一个Vue组件在使用前需要先被“注册”,这样Vue才能在渲染模板时找到其对应的实现。组件注册有两种方式:全局注册和局部注册...
- 焦虑!Vue3 组件频繁假死?6 个奇招让页面流畅度狂飙 500%!
-
前端圈的朋友们,谁还没在Vue3项目上踩过性能的坑?满心期待开发出的组件,一到高并发场景就频繁假死,用户反馈页面点不动,产品经理追着问进度,自己调试到心态炸裂!别以为这是个例,不少人在电商大促、数...
- 前端流行框架Vue3教程:26. 异步组件
-
根据上节课的代码,我们在切换到B组件的时候,发现并没有网络请求:异步组件:...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- idea eval reset (50)
- vue dispatch (70)
- update canceled (42)
- order by asc (53)
- spring gateway (67)
- 简单代码编程 贪吃蛇 (40)
- transforms.resize (33)
- redisson trylock (35)
- 卸载node (35)
- np.reshape (33)
- torch.arange (34)
- npm 源 (35)
- vue3 deep (35)
- win10 ssh (35)
- vue foreach (34)
- idea设置编码为utf8 (35)
- vue 数组添加元素 (34)
- std find (34)
- tablefield注解用途 (35)
- python str转json (34)
- java websocket客户端 (34)
- tensor.view (34)
- java jackson (34)
- vmware17pro最新密钥 (34)
- mysql单表最大数据量 (35)