vue框架下组件间通信的方式有多少,操作是什么
Admin 2022-06-30 群英技术资讯 340 次浏览
一个组件里面引入另外一个组件,此时构成了一种“父子关系”,当前组件为“父”,引入的组件为“子”,如当前组件(父),在父组件中通过 “:message” 向子组件通信。
<template> <div class="parent-box"> <div> <div>我是父页面</div> <div>{{message}}</div> </div> <children :message="toChildrenMsg"></children> </div> </template> <script> import Children from './Children.vue' //当前页引入子组件 export default { name:"Parent", components:{ Children }, data(){ return { message:'我是父页面的内容', toChildrenMsg:'从父页面传过到子页面的内容' } } } </script>
在子组件通过props进行接收,注意子组件props里面接收的对象名称必须与父组件中在子组件绑定的名称一致,当前例子为“message”,可以在组件return中this.的方式使用props里面的值。
<template> <div class="children-box"> <div> <div>我是子页面</div> <div>{{message}}</div> </div> </div> </template> <script> export default { name:"Children", props:{ message:{ type:String, //类型判断 default:'' //默认值 } } } </script>
子组件接收到父组件传过来的内容,实现效果如下图所示:
在子组件中通过this.$emit()方法向父组件通信,如下,点击触发事件,执行this.$emit('fromChildMethod'),触发父组件的fromChildMethod方法。
<template> <div class="children-box"> <div> <div>我是子页面</div> <div>{{message}}</div> <div><span @click="toParentMethod">点击触发父页面事件</span></div> </div> </div> </template> <script> export default { name:"Children", props:{ message:{ type:String, default:'' } }, methods:{ toParentMethod(){ this.$emit('fromChildMethod') } } } </script>
在父组件的子组件上绑定fromChildMethod方法,对该方法进行监听,当该方法触发时,执行父组件中相应的方法fromChild。
<template> <div class="parent-box"> <div> <div>我是父页面</div> <div style="font-size:12px;">{{message}}</div> <div style="font-size:12px;color:red">{{fromChildMsg}}</div> </div> <children :message="toChildrenMsg" @fromChildMethod="fromChild"></children> </div> </template> <script> import Children from './Children.vue' export default { name:"Parent", components:{ Children }, data(){ return { message:'我是父页面的内容', toChildrenMsg:'从父页面传过到子页面的内容', fromChildMsg:'' } }, methods:{ fromChild(){ this.fromChildMsg = '子页面触发的方法' //监听到子组件触发的方法,显示该内容 } } } </script>
当点击子组件的对应的span,触发方法,向父组件进行通知。
小结:父传子,props;子传父,this.$emit();触发、监听名称须一致。
真实的场景中,组件不仅仅是“父子”关系,还有“兄弟”关系跟跨层级组件等等。这时候props跟$emit可能就不太适用了,这时候它出现了,那就是Bus(事件总线),父子组件同样适用。
Bus之触发$emit、监听$on、关闭$off,主要用到的就是$emit跟$on。
先在项目中新建一个文件夹bus,里面有个index.js文件,创建一个新的Vue实例,然后导出模块。
接下来import这个新的Vue实例,也就是bus,常用的两种导入方式,一种是全局导入,另外一种是局部导入(需每个组件都导入一次)。以下为全局导入,在main.js里面将该bus作为当前Vue实例的原型方法,能直接在各组件里面通过this.bus的方式调用。
import Vue from 'vue' import App from './App' import bus from './bus/index' Vue.prototype.bus = bus new Vue({ el: '#app', components: { App }, template: '<App/>' })
下面展示实现bus通信过程,场景为父子,同样的,兄弟、跨层级用法与其类似:
Parent组件中向Children组件通信,通过this.bus.$emit()触发
<template> <div class="parent-box"> <div> <div>我是父页面</div> <div @click="toChildBus"><span>向子组件通信</span></div> </div> <children></children> </div> </template> <script> import Children from './Children.vue' export default { name:"Parent", components:{ Children }, methods:{ toChildBus(){ let val = '父组件向子组件通信' this.bus.$emit('toChild',val) //val为传过去的值,非必传 } } } </script>
Children组件监听Parent组件触发的事件(在mounted阶段进行绑定监听),注意事件名称要一致,通过this.bus.$on()监听,当总线中监听到触发该方法,拿到传过来的值(也可以在里面执行自定义方法)。
<template> <div class="children-box"> <div> <div>我是子页面</div> <div style="font-size:12px;color:blue;">{{fromParentMsg}}</div> </div> </div> </template> <script> export default { name:"Children", data(){ return { fromParentMsg:'' } }, mounted(){ this.bus.$off('toChild') this.bus.$on('toChild',val=>{ this.fromParentMsg = val //此处为复制操作,也可在里面执行相应的方法 }) } } </script>
效果图:
总结:父子,兄弟,跨级(祖孙等)通信写法相同,就不一一举例了,都是通过this.bus.$emit()触发,通过this.bus.$on()监听,执行相应的操作,切记:触发、监听名称必须相同!
Vuex相当于一个仓库,你可以往仓库里面放一些东西,保持存进去的时的状态,可以修改,也可以在需要的时候取出,是一个全局状态。本次只讲如何使用vuex进行通信,不深究其原理。
安装vuex
npm install vuex --save
这里我新建一个文件夹,名称为store,里面有一个index.js文件,创建一个Vuex.Store实例,然后导出这个实例,从图中可以明确看出store的大致结构及其要素,具体不展开讲,关于vuex的相关文章数不胜数,可以自行去了解,这里主要讲大致用法。
在mian.js全局引入,之后就可以直接使用了。
import Vue from 'vue' import App from './App' import router from './router' import bus from './bus/index' import store from './store/index' Vue.config.productionTip = false Vue.prototype.bus = bus new Vue({ el: '#app', router, store, components: { App }, template: '<App/>' })
方式一,this.$store.state.xxx,直接对state进行操作,在一个组件mounted阶段将值存如store中,当然也可在你想在的方法中进行操作。
<template> <div class="parent-box"> <div> <div>我是父页面</div> </div> <children></children> </div> </template> <script> import Children from './Children.vue' export default { name:"Parent", components:{ Children }, data(){ return { fromChildMsg:'' } } mounted(){ this.$store.state.msg = '父组件存入' //在此处通过方式一存起来 } } </script>
其他组件从store中取出,当然同样也可以进行修改。
<template> <div class="children-box"> <div> <div>我是子页面</div> <div @click="fromStore"><span>从store里面取</span></div> <div>{{fromStoreMsg}}</div> </div> </div> </template> <script> export default { name:"Children", data(){ return { fromStoreMsg:'' } }, methods:{ fromStore(){ this.fromStoreMsg = this.$store.state.msg } } } </script>
效果图:
方式二,通过this.$store.getters.xxx、mapGetters进行取出。
// store/index.js getters:{ getMsg:state=>{ return state.msg } }, //组件中取 this.$store.getters.getMsg //也可以用mapGetters的方式 import { mapGetters } from 'vuex' computed: { ...mapGetters(['getMsg']) },
对store存入数据该可以用mutations、actions(可异步)进行存入,具体就不展开了,有兴趣可以自己去深究。
可以通过动态路由、路由跳转方式进行传值,如this.$router.push({path:'xxx',query:{value:'xxx'}}),在跳转的时候顺便传值,通过this.$route.params.value和this.$route.query.value获取到传过来的参数。该方式有局限性,只能在相互跳转的组件通信取值,且直接在跳转之后的页面进行刷新取不到值,视情况而用。
sessionStorage、localStorage、cookie
多个组件之间的通信除了可以用bus、store之外,还比较一种常用的方式--缓存,在同一个窗口不关闭的情况下,该窗口下的其他组件都可以取到缓存中已经存好的值,利用sessionStorage.setItem(key,value)、localStorage.setItem(key,value)等将值存起来,其他组件可以通过sessionStorage.getItem(key)、localStorage.getItem(key)等方式拿到,多个页面共享缓存数据,刷新页面数据不会销毁,可以用sessionStorage.removeItem(key)、localStorage.removeItem(key)的方式将缓存移除,可用场景还是比较多的。
总结:大致介绍vue组件中几种常用的通信、传值方式,考虑不同的场景使用不同的方式,提高开发效率,减少bug的产生。
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。
猜你喜欢
从零开始在vue-cli4配置自适应vw布局的实现, 简介<br /> viewportWidth也是vw布局从配置上来说比rem布局简洁了很多,bu需要配置安装lib,也不需要增加一个rem.js文件<br /> 简称拎包使用<br /> 安装包<br /> <br /> <br /> npm install postcss-px-t
这篇文章主要为大家介绍了Vue2 中的数据劫持简写示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
javascript数组中的map方法和filter方法,下文有实例供大家参考,对大家了解操作过程或相关知识有一定的帮助,而且实用性强,希望这篇文章能帮助大家,下面我们一起来了解看看吧。
怎样用vue实现一个无缝的轮播效果?对于轮播效果的应用场景有很多,我们经常能在网站的头部看到,轮播功能也是比较实用的,但是一些新手在刚接触轮播时,实现的轮播效果有缝隙,效果并不好看,对此下面小编就给大家分享实现无缝轮播效果的代码。
本文主要介绍了解决React useEffect钩子带来的无限循环问题,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
成为群英会员,开启智能安全云计算之旅
立即注册Copyright © QY Network Company Ltd. All Rights Reserved. 2003-2020 群英 版权所有
增值电信经营许可证 : B1.B2-20140078 粤ICP备09006778号 域名注册商资质 粤 D3.1-20240008