Vue 传参踩坑之旅——事件总线与 props

Vue 传参踩坑之旅------事件总线与 props

缘由

今天突然发现项目出现了一个 bug,这里简单描述一下。

这里有 A、B、C、D 四个组件,关系为 A - 祖先、B - 父、C - 子、D - 叔(实际业务组件关系复杂很多)。

  • A - 祖先

    • B - 父

      • C - 子
    • D - 叔

现在 C 组件要用到 D 组件中的一个方法,由于组件关系比较复杂,我为了省事直接使用 mitt 进行自定义事件的触发,但是 D 组件又可能有多个(通过 v-for 生成的),此时就出现了问题,当存在多个 D 组件时,在 C 组件中调用了 D 中的方法会同时触发所有 D 组件的该事件。

有的小伙伴看完就知道问题所在了,没明白的也没关系,下面我会结合代码进行详细说明。

代码详细说明

先说明一下目录结构:

css 复制代码
 - App.vue
 - A.vue
 - B.vue
 - C.vue
 - D.vue
 - mitt.js
xml 复制代码
 <!-- App.vue -->
 <template>
   <A v-for="item of 2" :key="item"></A>
 </template>
 ​
 <script setup>
 import A from "./A.vue";
 </script>
xml 复制代码
 <!-- A.vue -->
 <template>
   <B></B>
   <D></D>
 </template>
 ​
 <script setup>
 import B from "./B.vue";
 import D from "./D.vue";
 </script>
xml 复制代码
 <!-- B.vue -->
 <template>
   <C></C>
 </template>
 ​
 <script setup>
 import C from "./C.vue";
 </script>
xml 复制代码
 <!-- C.vue -->
 <template>
   <button @click="handleDFunction">点击触发D组件方法</button>
 </template>
 ​
 <script setup>
 import mitt from "./mitt";
 ​
 function handleDFunction() {
   mitt.emit("handled");
 }
 </script>
xml 复制代码
 <!-- D.vue -->
 <template></template>
 ​
 <script setup>
 import mitt from "./mitt";
 ​
 function handled() {
   console.log("D组件方法被触发");
 }
 mitt.on("handled", handled);
 </script>
javascript 复制代码
 import mitt from 'mitt'
 ​
 export default new mitt()

以上就是精简后的示例代码,可以简单理解为兄弟组件传参。

效果展示

此时页面会显示两个 <button> 按钮。

当随便点击了一个后会输出两次 "D组件方法被触发"。

我们想要的效果是只会触发当前兄弟组件的事件,而此时触发了所有 D 组件的该事件。

问题分析

这个 bug 的原因其实也很简单,这是由于 mitt.on 注册同名事件会在数组后面追加,而非覆盖或者其他思路,简单通过代码说明一下它的源码实现:

scss 复制代码
 function mitt() {
   const eventMap = new Map(); // 事件表
   return {
     // 事件注册
     on(key, callback) {
       // 获取当前名称的事件列表
       const curEventList = eventMap.get(key)
       
       // 不是第一次注册该名称的事件,追加到数组最后
       if (curEventList) {
         curEventList.push(callback)
       }
       // 第一次注册该名称的事件,事件表中添加该事件列表
       else {
         eventMap.set(key, [callback])
       }
     },
     // 事件派发
     emit(key, ...args) {
       // 获取当前名称的事件列表
       const curEventList = eventList.get(key)
       
       // 存在该事件,则遍历数组依次执行
       if (curEventList) {
         curEventList.forEach((curEvent) => {
           curEvent(args)
         })
       }
     },
   }
 }

现在就能知道为什么会出现这种问题了,因为每一个 D 组件都对该事件名称进行了注册,因此实际上该事件名称注册了 n 个该事件,所以当 emit 派发时就会依次执行。

解决的化只需要将 emit 替换为 props / emits 就解决了。

总结

这个 bug 其实给了我蛮多启发的,也能理解之前看到别人说业务组件中尽量使用 props / emits 了,事件总线的原理我其实是知道的,但是还是会产生这种 bug,因为随着业务的发展,新功能的产生以及老功能的优化等都可能对原来的数据结构或组件结构进行更改,在我们写业务代码时是比较难精准的预料到未来的变化的,此时就需要我们的代码有足够高的容错性,诸如 provide / inject、事件总线等组件通信方式固然很方便,但是在代码重构或其他一些比较重大改动时项目的数据流又会变得难以预测。

props / emits 的缺点:

  • 代码量会变得更多、冗余

props / emits 的优点:

  • 组件数据流变得清晰
  • 当项目结构发生变化时重构会更简单
  • 不容易产生奇怪的问题
相关推荐
Amd7946 分钟前
Nuxt.js 应用中的 webpack:compiled 事件钩子
前端·webpack·开发·编译·nuxt.js·事件·钩子
生椰拿铁You14 分钟前
09 —— Webpack搭建开发环境
前端·webpack·node.js
狸克先生26 分钟前
如何用AI写小说(二):Gradio 超简单的网页前端交互
前端·人工智能·chatgpt·交互
baiduopenmap40 分钟前
百度世界2024精选公开课:基于地图智能体的导航出行AI应用创新实践
前端·人工智能·百度地图
loooseFish1 小时前
小程序webview我爱死你了 小程序webview和H5通讯
前端
菜牙买菜1 小时前
让安卓也能玩出Element-Plus的表格效果
前端
请叫我欧皇i1 小时前
html本地离线引入vant和vue2(详细步骤)
开发语言·前端·javascript
533_1 小时前
[vue] 深拷贝 lodash cloneDeep
前端·javascript·vue.js
guokanglun1 小时前
空间数据存储格式GeoJSON
前端
zhang-zan2 小时前
nodejs操作selenium-webdriver
前端·javascript·selenium