vue中非父子组件通信

x33g5p2x  于2022-02-12 转载在 Vue.js  
字(4.3k)|赞(0)|评价(0)|浏览(469)

在上一篇博客中,我们总结了父子组件通信方式有:props + emit。这里我们将总结一下,非父子组件通信方式,这里还不涉及到Vuex。
如果存在祖孙组件,我们可以通过ProvideInject进行通信。
如果不是祖孙组件,也不是父子组件,我们可以采用Mitt全局事件总线第三方库来实现。
一、Provide和Inject
ProvideInject用于非父子组件之间共享数据,比如有一些深度嵌套的组件,子组件想要获取父组件的数据,如果不存在Provide和Inject选项,我们可以通过props进行一次传递数据,但是这样做太过于繁琐。
对于上述情况,我们可以使用ProvideInject
无论组件结构嵌套有多深,父组件都可以作为子组件数据的提供者。
父组件存在Provide来提供数据。
子组件存在Inject来获取数据。
在实际过程中,父组件不知道哪个子组件使用其数据,子组件也不知道使用的是哪个父组件的数据。
Provide和Inject的基本使用

  1. //父组件
  2. <template>
  3. <div>
  4. <Content></Content>
  5. </div>
  6. </template>
  7. <script>
  8. import Content from "./components/Content.vue";
  9. export default {
  10. data() {
  11. return {};
  12. },
  13. provide:{
  14. name:"张三",
  15. age:20
  16. },
  17. components: {
  18. Content
  19. }
  20. };
  21. </script>
  22. <style scoped></style>
  1. //子组件
  2. <template>
  3. <div>
  4. <h1>这里是Content组件</h1>
  5. <h1>{{ name }} -- {{age}}</h1>
  6. <ContentBase></ContentBase>
  7. </div>
  8. </template>
  9. <script>
  10. import ContentBase from "./ContentBase.vue";
  11. export default {
  12. data() {
  13. return {};
  14. },
  15. components: {
  16. ContentBase
  17. },
  18. inject:["name", "age"]
  19. };
  20. </script>
  21. <style scoped></style>
  1. //孙组件
  2. <template>
  3. <div>
  4. <h1>这里是contentBase组件</h1>
  5. <h1>{{name}} -- {{age}}</h1>
  6. </div>
  7. </template>
  8. <script>
  9. export default {
  10. data() {
  11. return {};
  12. },
  13. inject: ["name", "age"]
  14. };
  15. </script>
  16. <style scoped>
  17. </style>

最终显示结果为:

二、Provide和Inject的另一种写法
我们思考一种情况,如果Provide拿到的数据,是从data拿到的数据,此时如果获取? 此时this可以使用吗?

  1. <template>
  2. <div>
  3. <Content></Content>
  4. </div>
  5. </template>
  6. <script>
  7. import Content from "./components/Content.vue";
  8. export default {
  9. data() {
  10. return {
  11. source: ["111","222", "333"]
  12. };
  13. },
  14. provide:{
  15. name:"张三",
  16. age:20,
  17. res: this.source.length //我们在此时增加res,想要通过this.source.length拿到数组的长度
  18. },
  19. components: {
  20. Content
  21. }
  22. };
  23. </script>
  24. <style scoped></style>

该结果是错误的。

报错信息显示,我们从undefined上读取属性,此时this为undefined。原因:从上面代码我们可以看出,this指向的是undefiend(因为this执行最外层)。
解决方法
我们将Provide设置为一个函数,并且返回一个对象,如下代码所示:

  1. <template>
  2. <div>
  3. <Content></Content>
  4. </div>
  5. </template>
  6. <script>
  7. import Content from "./components/Content.vue";
  8. export default {
  9. data() {
  10. return {
  11. source: ["111", "222", "333"],
  12. };
  13. },
  14. provide() {
  15. return {
  16. name: "张三",
  17. age: 20,
  18. res: this.source.length,
  19. };
  20. },
  21. components: {
  22. Content,
  23. },
  24. };
  25. </script>
  26. <style scoped></style>

显示结果:

此时我们再思考一个问题,如果我们向data数组中新增一个元素,在其他地方获取的数组长度会跟随变化吗?

  1. <template>
  2. <div>
  3. <Content></Content>
  4. <button @click="addOneItem">点击</button>
  5. </div>
  6. </template>
  7. <script>
  8. import Content from "./components/Content.vue";
  9. export default {
  10. data() {
  11. return {
  12. source: ["111", "222", "333"],
  13. };
  14. },
  15. provide() {
  16. return {
  17. name: "张三",
  18. age: 20,
  19. res: this.source.length,
  20. };
  21. },
  22. components: {
  23. Content,
  24. },
  25. methods:{ //在这里添加点击事件
  26. addOneItem() {
  27. this.source.push("nnn")
  28. console.log(this.source)
  29. }
  30. }
  31. };
  32. </script>
  33. <style scoped></style>

结果如上图所示,可以看到数据是被添加进去的,但是子组件并没有检测到数据的变化。
此时我们可以使用computed来检测this.source.length的变化,使用代码如下所示。

  1. <template>
  2. <div>
  3. <Content></Content>
  4. <button @click="addOneItem">点击</button>
  5. </div>
  6. </template>
  7. <script>
  8. import Content from "./components/Content.vue";
  9. import { computed } from "vue" //从vue中引入computed
  10. export default {
  11. data() {
  12. return {
  13. source: ["111", "222", "333"],
  14. };
  15. },
  16. provide() {
  17. return {
  18. name: "张三",
  19. age: 20,
  20. res: computed(() => this.source.length), //在这里添加computed
  21. };
  22. },
  23. components: {
  24. Content,
  25. },
  26. methods:{
  27. addOneItem() {
  28. this.source.push("nnn")
  29. console.log(this.source)
  30. }
  31. }
  32. };
  33. </script>
  34. <style scoped></style>

因为我们通过computed获取的是一个对象,此时我们通过value属性拿到值。
三、全局事件总线mitt库
在vue2时,如果我们使用事件总线可以使用this.$bus = new Vue()也就是实例化一个vue对象。但是我们在vue3中不能这样用。所以我们采用第三方库来实现组件之间的通信。这个第三方库为mitt
一、安装

  1. npm install mitt

在文件中引入并且进行初始化导出。

  1. import mitt from "mitt";
  2. const emitter = new mitt()
  3. export default emitter

监听事件,第一个参数是事件名,第二个参数是回调函数。

  1. emitter.on("why", (data) => {
  2. console.log(data)
  3. })
  4. //*表示可以监听全部的事件。
  5. emitter.on("*", (type, data) => {
  6. console.log(type, data)
  7. })

取消事件

  1. //取消emitter中所有的监听
  2. emitter.all.clear()
  3. //或者
  4. //定义一个函数
  5. function onFoo(){}
  6. emitter.on("foo", onFoo)
  7. emitter.on("foo", onFoo)

相关文章

最新文章

更多