前言

前期分享的​​测试开发系列!Vue 组件间通信方式汇总,总有一款适合你( 5分钟教程-附项目实战案例 )​​中介绍了 3 种组件间的通信方法,分别是:

  • props
  • 全局事件总线
  • 消息订阅与发布

今天给大家分享第 4 种组件间的通信方法:​​自定义事件​

适用于:​子组件给父组件传递数据​

使用方法

例如:​​Student.vue​​​组件是​​School.vue​​​组件的子组件, 子组件​​Student.vue​​​想要将学生姓名传递给父组件​​School.vue​

第1种方法:ref标识

​School.vue组件内容如下​

<template> <div> <h1>学生姓名是:{{studentName}}</h1> <!-- 通过父组件给子组件绑定一个自定义事件实现:子给父传递数据(第1种写法:使用ref) --> <Student ref="student" @click.native="show"/> </div> </template> <script> import Student from "./components/Student" export default { name:'School', components:{Student}, data(){ return { studentName:'' } }, methods: { getStudentName(name,...params){ //使用name形参接收Student组件中传递过来的实参name,...params用于接收剩余的参数 this.studentName = name } }, mounted() { //先在methods中准备好getStudentName方法 this.$refs.student.$on('studentName', this.getStudentName) //绑定自定义事件,指定回调函数为getStudentName } } </script>

​Student.vue组件内容如下​

<template> <div> <button @click="sendStudentName">发送学生名给School组件</button> </div> </template> <script> export default { name:'Student', data() { return { name: '刘浩' } }, methods: { sendStudentName(){ //触发School组件实例身上的studentName事件,把学生名给School组件,再传递2个额外的参数:666,777 this.$emit('studentName', this.name, 666, 777) } } } </script>

第2种方法:v-on或@

​School.vue组件内容如下​

<template> <div> <h1>学生姓名是:{{studentName}}</h1> <!-- 通过父组件给子组件绑定一个自定义事件实现:子给父传递数据(第2种写法:使用@或v-on) --> <Student v-on:studentName="getStudentName"/> <!-- 简写形式 --> <!--<Student @studentName="getStudentName"/>--> </div> </template> <script> import Student from "./components/Student" export default { name:'School', components:{Student}, data(){ return { studentName:'' } }, methods: { getStudentName(name,...params){ //使用name形参接收Student组件中传递过来的实参name,...params用于接收剩余的参数 this.studentName = name } } } </script>

​Student.vue组件内容如下​

<template> <div> <button @click="sendStudentName">发送学生名给School组件</button> </div> </template> <script> export default { name:'Student', data() { return { name: '刘浩' } }, methods: { sendStudentName(){ //触发School组件实例身上的studentName事件,把学生名给School组件,再传递2个额外的参数:666,777 this.$emit('studentName', this.name, 666, 777) } } } </script>

项目实战

在 ​​200行纯前端Vue代码!教你写一个专属TodoList【零基础友好】
​​项目实战案例中,使用的​​props​​组件间通信方式,这种通信方式有一些不足之处。

在​​测试开发系列!Vue 组件间通信方式汇总,总有一款适合你( 5分钟教程-附项目实战案例 )​​一文中有详细描述该种通信方式的不足之处,同时也介绍了其他 2 种组件间通信方式,不妨点开链接重新温习一下。

今天介绍的​​自定义事件​​​的组件间通信方式的 2 种写法,可以很好的替换 ​​todoList案例​​中所使用的通信方式。

​父组件:App.vue组件内容如下​

<template> <div> <ListFooter :todos="todos" @checkAllTodo="checkAllTodo" @clearAllTodo="clearAllTodo"/> </div> </template> <script> import ListFooter from "./components/ListFooter" export default { name:'App', components:{ListFooter}, data(){ return { //由于todos是ListHeader组件和ListFooter组件都在使用,所以放在App中(状态提升) todos:JSON.parse(localStorage.getItem('todos')) || [] } }, methods: { //清除所有已经完成的todo clearAllTodo(){ this.todos = this.todos.filter((todo)=>{ return !todo.done }) } } } </script>

这里的​​App.vue​​​组件就是父组件,​​ListFooter.vue​​​组件是子组件,以​​clearAllTodo自定义事件​​为例。

使用的是第2种方法:v-on的简写形式@clearAllTodo="clearAllTodo",​​@​​​后的​​clearAllTodo​​是自定义事件名称。

子组件​​ListFooter.vue​​​在传递数据时只需要触发这个事件名称就可以传递数据给父组件​​App.vue​​​,​​=​​​后的​​clearAllTodo​​​是​​methods 回调函数​​的名称。

​子组件:ListFooter.vue组件内容如下​

<template> <div> <button @click="clearAll">清除已完成任务</button> </div> </template> <script> export default { name:'ListFooter', methods:{ clearAll(){ this.$emit('clearAllTodo') //使用自定义事件clearAllTodo实现---App组件中给绑定的自定义事件 } } } </script>

在子组件​​ListFooter.vue​​​中使用​​$emit​​​来触发父组件​​App.vue​​​中自定义事件​​clearAllTodo​​的逻辑。

小结

  1. ​自定义事件​​​是一种组件间通信的方式,适用于:​​子组件传递数据给父组件​​。
  2. 使用场景:A是父组件,B是子组件,B想给A传数据,需要在 A 中给 B 绑定自定义事件,事件的回调在 A 中定义。
  3. 绑定自定义事件:
    (1).第 1 种方式,在父组件​​App.vue​​中:
<template>
<div>
<Student @studentName="sendStudentName"/>
<!--或者使用以下非简写方式-->
<Student v-on:studentName="sendStudentName"/>
</div>
</template>

(2).第 2 种方式,在父组件​​App.vue​​中:

<template>
<div>
<Student ref="student"/>
</div>
</template>
<script>
import ListFooter from "./components/ListFooter"

export default {
name:'App',
components:{ListFooter},
data(){
return {
studentName:'刘浩'
}
},
methods: {
getStudentName(name,...params){
this.studentName = name
}
},
mounted(){
this.$refs.student.$on('studentName',this.getStudentName)//提前准备好getStudentName回调方法
}
}
</script>
  1. 子组件中触发父组件中的自定义事件:
    ​​​this.$emit('studentName',传递的数据)​
  2. 子组件中解绑自定义事件:
    ​​​this.$off('studentName')​
  3. 组件上也可以绑定​​原生 DOM 事件​​​,需要使用​​native​​修饰符。

欢迎关注 ​无量测试之道​​ 公众号,回复​​领取资源​

Python+Unittest框架API自动化、

Python+Unittest框架API自动化、

Python+Pytest框架API自动化、

Python+Pandas+Pyecharts大数据分析、

Python+Selenium框架Web的UI自动化、

Python+Appium框架APP的UI自动化、

Python编程学习资源干货、

Vue前端组件化框架开发、

资源和代码 免费送啦~
文章下方有公众号二维码,可直接微信扫一扫关注即可。

备注:我的个人公众号已正式开通,致力于IT互联网技术的分享。

包含:数据分析、大数据、机器学习、测试开发、API接口自动化、测试运维、UI自动化、性能测试、代码检测、编程技术等。

微信搜索公众号:​无量测试之道​​​,或扫描下方二维码:

Vue 组件间通信方式:自定义事件_传递数据

添加关注,让我们一起共同成长!