vue3源码分析——实现组件通信provide,inject

简介: 问题解决: 问题1和问题2都很好解决,对外导出函数,传递对应的参数,只是数据存储在哪里的问题,经过仔细的思考,会发现,组件的数据是需要进行共享的,父组件存入的数据,里面的所有子组件和孙子组件都可以共享,那么存储在实例上,是不是一个不错的选择呢? inject 是获取父级组件的数据,那么在实列上还需要传入parent

引言


<<往期回顾>>


1.vue3源码分析——rollup打包monorepo

2.vue3源码分析——实现组件的挂载流程

3.vue3源码分析——实现props,emit,事件处理等

4.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);
  })


分析


在上面的测试拥立中,可以得到以下内容:


1.getCurrentInstance只能在setup函数中使用

2.对外导出的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>')
  })


分析


从上面的测试用例中进行需求分析,


1.provide api是需要有两个参数,一个key,另一个是value, 有点类似与sessionStorage这种set值的方式


2.inject api则是只需要一个key,来进行get操作


3.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>')
  })


分析


上面的测试用例相对于父子组件的测试用例来说,增加了一个孙子组件。


1.孙子(Child组件) 和 父亲(Foo组件) 都可以获取 爷爷(App组件) 的值

2.其他的没啥变化


问题解决: 想要让孙子组件获取爷爷组件的数据,那是否可以让父组件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组件提供数据,其他只是嵌套,不提供数据。存在下面问题:


1.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是否存在,不存在则往上查找

相关文章
|
10天前
|
JavaScript API 数据处理
vue3使用pinia中的actions,需要调用接口的话
通过上述步骤,您可以在Vue 3中使用Pinia和actions来管理状态并调用API接口。Pinia的简洁设计使得状态管理和异步操作更加直观和易于维护。无论是安装配置、创建Store还是在组件中使用Store,都能轻松实现高效的状态管理和数据处理。
38 3
|
2月前
|
存储 JavaScript 数据管理
除了provide/inject,Vue3中还有哪些方式可以避免v-model的循环引用?
需要注意的是,在实际开发中,应根据具体的项目需求和组件结构来选择合适的方式来避免`v-model`的循环引用。同时,要综合考虑代码的可读性、可维护性和性能等因素,以确保系统的稳定和高效运行。
33 1
|
2月前
|
JavaScript
Vue3中使用provide/inject来避免v-model的循环引用
`provide`和`inject`是 Vue 3 中非常有用的特性,在处理一些复杂的组件间通信问题时,可以提供一种灵活的解决方案。通过合理使用它们,可以帮助我们更好地避免`v-model`的循环引用问题,提高代码的质量和可维护性。
42 1
|
4天前
|
JavaScript
vue使用iconfont图标
vue使用iconfont图标
38 1
|
14天前
|
JavaScript 关系型数据库 MySQL
基于VUE的校园二手交易平台系统设计与实现毕业设计论文模板
基于Vue的校园二手交易平台是一款专为校园用户设计的在线交易系统,提供简洁高效、安全可靠的二手商品买卖环境。平台利用Vue框架的响应式数据绑定和组件化特性,实现用户友好的界面,方便商品浏览、发布与管理。该系统采用Node.js、MySQL及B/S架构,确保稳定性和多功能模块设计,涵盖管理员和用户功能模块,促进物品循环使用,降低开销,提升环保意识,助力绿色校园文化建设。
|
2月前
|
JavaScript 前端开发 开发者
vue学习第一章
欢迎来到我的博客!我是瑞雨溪,一名热爱前端的大一学生,专注于JavaScript与Vue,正向全栈进发。博客分享Vue学习心得、命令式与声明式编程对比、列表展示及计数器案例等。关注我,持续更新中!🎉🎉🎉
46 1
vue学习第一章
|
2月前
|
JavaScript 前端开发 索引
vue学习第三章
欢迎来到瑞雨溪的博客,一名热爱JavaScript与Vue的大一学生。本文介绍了Vue中的v-bind指令,包括基本使用、动态绑定class及style等,希望能为你的前端学习之路提供帮助。持续关注,更多精彩内容即将呈现!🎉🎉🎉
32 1
|
2月前
|
缓存 JavaScript 前端开发
vue学习第四章
欢迎来到我的博客!我是瑞雨溪,一名热爱JavaScript与Vue的大一学生。本文介绍了Vue中计算属性的基本与复杂使用、setter/getter、与methods的对比及与侦听器的总结。如果你觉得有用,请关注我,将持续更新更多优质内容!🎉🎉🎉
39 1
vue学习第四章
|
2月前
|
JavaScript 前端开发 算法
vue学习第7章(循环)
欢迎来到瑞雨溪的博客,一名热爱JavaScript和Vue的大一学生。本文介绍了Vue中的v-for指令,包括遍历数组和对象、使用key以及数组的响应式方法等内容,并附有综合练习实例。关注我,将持续更新更多优质文章!🎉🎉🎉
30 1
vue学习第7章(循环)
|
2月前
|
JavaScript 前端开发
vue学习第九章(v-model)
欢迎来到我的博客,我是瑞雨溪,一名热爱JavaScript与Vue的大一学生,自学前端2年半,正向全栈进发。此篇介绍v-model在不同表单元素中的应用及修饰符的使用,希望能对你有所帮助。关注我,持续更新中!🎉🎉🎉
34 1
vue学习第九章(v-model)

热门文章

最新文章