vue3源码分析——实现组件通信provide,inject
ztj100 2024-11-12 14:22 26 浏览 0 评论
引言
<<往期回顾>>
- vue3源码分析——rollup打包monorepo
- vue3源码分析——实现组件的挂载流程
- vue3源码分析——实现props,emit,事件处理等
- vue3源码分析——实现slots
本期来实现, vue3组件通信的provide,inject,所有的源码请查看
getCurrentInstance
在实现provide/inject之前,先来实现getCurrentInstance,由于在provide/inject中会使用到这个api,在开发的时候,这个api使用的频率也是挺频繁的。
getCurrentInstance 是获取当前组件的实列,并且只能在setup函数中使用
测试用例
test('test getCurrentInstance', () => {
const Foo = {
name: 'Foo',
setup() {
// 获取子组件的实列,并且期望是子组件的名称是Foo
const instance = getCurrentInstance();
expect(instance.type.name).toBe('Foo');
return {
count: 1
}
},
render() {
return h('div', {}, '122')
}
}
const app = createApp({
name: 'App',
setup() {
// 获取父组件的实例,期待父组件的名称是定义的App
const instance = getCurrentInstance();
expect(instance.type.name).toBe('App');
return {
count: 2
}
},
render() {
return h('div', { class: 'container' }, [h(Foo, {}, {})])
}
})
// 挂载组件
const appDoc = document.querySelector('#app')
app.mount(appDoc);
})
分析
在上面的测试拥立中,可以得到以下内容:
- getCurrentInstance只能在setup函数中使用
- 对外导出的api,获取的是当前组件的实列
问题解决:
对于上面两个问题,只需要导出一个函数,并且在全局定义一个变量,在setup执行的时候,赋值全局变量即可拿到当前组件的实例,然后setup执行之后,清空即可
编码
// setup执行是在setupStatefulComponent函数中执行的,来进行改造
// 定义全局的变量,存储当前实例
let currentInstance = null;
function setupStatefulComponent(instance: any) {
// ……省略其他
// 获取组件的setup
const { setup } = Component;
if (setup) {
currentInstance = instance
const setupResult = setup(shallowReadonly(instance.props), { emit: instance.emit })
// 情况操作
currentInstance = null
}
// ……省略其他
}
// 对外导出函数,提供全局的api
export function getCurrentInstance() {
return currentInstance
}
getCurrentInstance 有没有想到实现方式这么简单哇!
provide/inject
provide和inject需要配套使用才方便用于测试,这里就从功能分析,来逐步完成这两个api.
父子组件传值
父子组件传值可以使用props/emit来实现,还记得是怎么实现的么?
测试用例
test('test provide basic use', () => {
const Foo = {
name: 'Foo',
setup() {
// 子组件接受数据
const count = inject('count')
const str = inject('str')
return {
count,
str
}
},
render() {
return h('div', {}, this.str + this.count)
}
}
const app = createApp({
name: 'App',
setup() {
// 父组件提供数据,
provide('count', 1);
provide('str', 'str');
},
render() {
return h('div', { class: 'container' }, [h(Foo, {})])
}
})
const appDoc = document.querySelector('#app')
app.mount(appDoc);
const container = document.querySelector('.container') as HTMLElement;
expect(container.innerHTML).toBe('<div>str1</div>')
})
分析
从上面的测试用例中进行需求分析,
- provide api是需要有两个参数,一个key,另一个是value, 有点类似与sessionStorage这种set值的方式
- inject api则是只需要一个key,来进行get操作
- provide存的数据,存在哪里呢?
问题解决: 问题1和问题2都很好解决,对外导出函数,传递对应的参数,只是数据存储在哪里的问题,经过仔细的思考,会发现,组件的数据是需要进行共享的,父组件存入的数据,里面的所有子组件和孙子组件都可以共享,那么存储在实例上,是不是一个不错的选择呢? inject 是获取父级组件的数据,那么在实列上还需要传入parent
编码
由于需要在实例上存储provide,首先就在createInstance中的实例,在初始化就赋值
export function createComponentInstance(vnode, parent) {
const instance = {
// ……省略其他属性
// 提供数据
provides: {},
parent,
}
return instance
}
// 有了实例,分别创建provide,inject函数
export function provide(key, val){
// 将数据存在实例上,先进行获取
const instance = getCurrentInstance();
if(instance){
instance.provides[key] = val
}
}
export function inject(key){
// 从实列上取值
const instance = getCurrentInstance();
if(instance){
// 获取父级provides
const provides = instance.parent?.provides;
if(key in provides){
return provides[key]
}
return null
}
}
一个简单的prvide/inject就实现啦,接下来进行需求升级,爷孙组件数据传递
爷孙组件传值
无可厚非,就是孙子组件需要从爷爷组件中获取值,父组件不提供数据
测试用例
test('test provide exit grandfather', () => {
const Child = {
name: 'Foo',
setup() {
// 孙子组件也可以取值
const count = inject('count')
const str = inject('str')
return {
count,
str
}
},
render() {
return h('div', {}, this.str + this.count)
}
}
const Father = {
name: 'Father',
setup() {
// 子组件可以取值
const count = inject('count')
return {
count
}
},
render() {
return h('div', {}, [h('p', {}, this.count), h(Child, {})])
}
}
const app = createApp({
name: 'App',
setup() {
// 爷爷提供数据
provide('count', 1);
provide('str', 'str');
return {}
},
render() {
return h('div', { class: 'container' }, [h(Father, {})])
}
})
const appDoc = document.querySelector('#app');
app.mount(appDoc);
const container = document.querySelector('.container') as HTMLElement;
expect(container.innerHTML).toBe('<div><p>1</p><div>str1</div></div>')
})
分析
上面的测试用例相对于父子组件的测试用例来说,增加了一个孙子组件。
- 孙子(Child组件) 和 父亲(Foo组件) 都可以获取 爷爷(App组件) 的值
- 其他的没啥变化
问题解决: 想要让孙子组件获取爷爷组件的数据,那是否可以让父组件Foo在初始化就获取他父组件App的provides
编码
// 需要在组件初始化的时候,获取父组件的数据,修改下初始化的内容
export function createComponentInstance(vnode, parent) {
const instance = {
// ……省略其他属性
// 存在则用,不存在还是空对象
provides: parent ? parent.provides : {},
parent,
}
return instance
}
是不是感觉非常简单哇,那接下来在升级下,inject获取provide的数据,需要就近原则来进行获取
就近原则获取数据
就近原则的意思是说,如果父组件有就拿父组件的,父组件没有就那爷爷组件的,爷爷组件没有继续往上找,直到找到跟组件App上,如果还没有就为null
测试用例
test('get value by proximity principle(就近原则) ', () => {
// 孙子组件来获取数据
const GrandSon = {
name: 'GrandSon',
setup() {
const count = inject('count')
const str = inject('str')
return {
count,
str
}
},
render() {
return h('div', {}, this.str + this.count)
}
}
// 子组件提供count
const Child = {
name: 'Child',
setup() {
provide('count', 100)
},
render() {
return h(GrandSon)
}
}
// 父亲组件,不提供数据
const Father = {
name: 'Father',
render() {
return h(Child)
}
}
// 跟组件app,提供,count,str
const app = createApp({
name: 'App',
setup() {
provide('count', 1);
provide('str', 'str');
return {}
},
render() {
return h('div', { class: 'container' }, [h(Father, {})])
}
})
// ……省略挂载
const container = document.querySelector('.container') as HTMLElement;
expect(container.innerHTML).toBe('<div>str100</div>')
})
分析
在上面的测试用例中,存在4个组件,只有app组件和Child组件提供数据,其他只是嵌套,不提供数据。存在下面问题:
- inject怎么去查找provides的数据,一层一层的查找
问题解决: 怎么查找呢,在inject里面递归? NO,换一个角度,inject查找数据的时候,是不是有点像原型链的方式来进行查找呢?YES,那就是需要在provide里面来构建一条原型链。
原型链, 啥叫做原型链呢?请查看
编码
// 只需要改造provide函数即可
export function provide(key, val) {
// 数据需要存储在当前的实例上面
const instance = getCurrentInstance();
if (instance) {
let { provides } = instance;
// 正对多层组件,需要把当前组件的__proto__绑定到父级上面,形成原型链,可以访问到最顶层的数据
const parentProvides = instance.parent && instance.parent.provide;
// 只有父级的provides和当前的provides是相同的时候为第一次调用provide,后续调用就不需要绑定原型了
if (parentProvides === provides) {
provides = instance.providers = Object.create(parentProvides || {});
}
provides[key] = val;
}
}
总结
本期主要完成了getCurrentInstance,provide,inject的实现,在getCurrentInstance中只是用了一个中间变量,而provide是把数据存在当前的instance当中,provide里面还用到了原型链的知识,通过原型的方式来查询key是否存在,不存在则往上查找
相关推荐
- sharding-jdbc实现`分库分表`与`读写分离`
-
一、前言本文将基于以下环境整合...
- 三分钟了解mysql中主键、外键、非空、唯一、默认约束是什么
-
在数据库中,数据表是数据库中最重要、最基本的操作对象,是数据存储的基本单位。数据表被定义为列的集合,数据在表中是按照行和列的格式来存储的。每一行代表一条唯一的记录,每一列代表记录中的一个域。...
- MySQL8行级锁_mysql如何加行级锁
-
MySQL8行级锁版本:8.0.34基本概念...
- mysql使用小技巧_mysql使用入门
-
1、MySQL中有许多很实用的函数,好好利用它们可以省去很多时间:group_concat()将取到的值用逗号连接,可以这么用:selectgroup_concat(distinctid)fr...
- MySQL/MariaDB中如何支持全部的Unicode?
-
永远不要在MySQL中使用utf8,并且始终使用utf8mb4。utf8mb4介绍MySQL/MariaDB中,utf8字符集并不是对Unicode的真正实现,即不是真正的UTF-8编码,因...
- 聊聊 MySQL Server 可执行注释,你懂了吗?
-
前言MySQLServer当前支持如下3种注释风格:...
- MySQL系列-源码编译安装(v5.7.34)
-
一、系统环境要求...
- MySQL的锁就锁住我啦!与腾讯大佬的技术交谈,是我小看它了
-
对酒当歌,人生几何!朝朝暮暮,唯有己脱。苦苦寻觅找工作之间,殊不知今日之事乃我心之痛,难道是我不配拥有工作嘛。自面试后他所谓的等待都过去一段时日,可惜在下京东上的小金库都要见低啦。每每想到不由心中一...
- MySQL字符问题_mysql中字符串的位置
-
中文写入乱码问题:我输入的中文编码是urf8的,建的库是urf8的,但是插入mysql总是乱码,一堆"???????????????????????"我用的是ibatis,终于找到原因了,我是这么解决...
- 深圳尚学堂:mysql基本sql语句大全(三)
-
数据开发-经典1.按姓氏笔画排序:Select*FromTableNameOrderByCustomerNameCollateChinese_PRC_Stroke_ci_as//从少...
- MySQL进行行级锁的?一会next-key锁,一会间隙锁,一会记录锁?
-
大家好,是不是很多人都对MySQL加行级锁的规则搞的迷迷糊糊,一会是next-key锁,一会是间隙锁,一会又是记录锁。坦白说,确实还挺复杂的,但是好在我找点了点规律,也知道如何如何用命令分析加...
- 一文讲清怎么利用Python Django实现Excel数据表的导入导出功能
-
摘要:Python作为一门简单易学且功能强大的编程语言,广受程序员、数据分析师和AI工程师的青睐。本文系统讲解了如何使用Python的Django框架结合openpyxl库实现Excel...
- 用DataX实现两个MySQL实例间的数据同步
-
DataXDataX使用Java实现。如果可以实现数据库实例之间准实时的...
- MySQL数据库知识_mysql数据库基础知识
-
MySQL是一种关系型数据库管理系统;那废话不多说,直接上自己以前学习整理文档:查看数据库命令:(1).查看存储过程状态:showprocedurestatus;(2).显示系统变量:show...
- 如何为MySQL中的JSON字段设置索引
-
背景MySQL在2015年中发布的5.7.8版本中首次引入了JSON数据类型。自此,它成了一种逃离严格列定义的方式,可以存储各种形状和大小的JSON文档,例如审计日志、配置信息、第三方数据包、用户自定...
你 发表评论:
欢迎- 一周热门
-
-
MySQL中这14个小玩意,让人眼前一亮!
-
旗舰机新标杆 OPPO Find X2系列正式发布 售价5499元起
-
【VueTorrent】一款吊炸天的qBittorrent主题,人人都可用
-
面试官:使用int类型做加减操作,是线程安全吗
-
C++编程知识:ToString()字符串转换你用正确了吗?
-
【Spring Boot】WebSocket 的 6 种集成方式
-
PyTorch 深度学习实战(26):多目标强化学习Multi-Objective RL
-
pytorch中的 scatter_()函数使用和详解
-
与 Java 17 相比,Java 21 究竟有多快?
-
基于TensorRT_LLM的大模型推理加速与OpenAI兼容服务优化
-
- 最近发表
- 标签列表
-
- 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)