博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
沃土前端社区Vue系列教程 - event bus 和 vuex
阅读量:5752 次
发布时间:2019-06-18

本文共 7324 字,大约阅读时间需要 24 分钟。

在开发过程中,父子组件传递数据,我们用props和$emit可以解决问题,那么非父子组件之间的数据传递我们要怎么解决呢,一般有两个方案,一个event bus(事件总线)和 vuex。我们先来说event bus。

一、用vue脚手架构建一个简单的项目

# 全局安装 vue-clinpm install --global vue-cli# 创建一个基于 webpack 模板的新项目vue init webpack vuexdemo# 安装依赖,走你cd vuexdemonpm installnpm run dev

二、使用event bus实现跨组件通信

event bus事件总线,组件之间的通信需要使用一个中介来实现,在event bus里面使用一个空的Vue实例来做为通信的桥梁,下面是使用event bus来实现改变登录状态的demo。

  1. 在原有main.js里加上这么一句
// 把busVue.prototype.bus = new Vue();
  1. 修改原有的Index.vue组件,添加如下代码:
data() {    return {      loginStatus: false    };  },  created() {    // 监听事件    this.bus.$on("login", (data) => {      this.loginStatus = data;    });  }

修改后的Index.vue代码如下:

  1. 新建组件Login.vue,在组件Login.vue添加登录操作,在登录成功的时候,触发‘login’事件,Login完整代码如下:
  1. 同时在vue-router里面配置路由/login,代码如下:
import Vue from 'vue'import Router from 'vue-router'import Index from '@/components/Index'import Login from '@/components/Login'Vue.use(Router)export default new Router({  routes: [    {      path: '/',      name: 'index',            component: Index    },    {      path: '/login',      name: 'login',      component: Login    }  ]})
  1. 测试:Index.vue与Login.vue是两个非父子关系的组件,当我们点击Login组件里面的立即登录的时候,Index组件里面显示的登录状态就会发生改变,这样就实现了非父子组件之间的通信。需要注意的是:负责监听的组件必须先挂载,才能使用$emit来触发事件。简单来讲,类似jQuery一样,事件必须先绑定,你才能取触发事件。

三、使用vuex实现跨组件通信

跨组件通信,使用event bus的做法,对于简单的应用来讲,但是对于复杂的应用来讲,event bus对于事件的管理就可能力不从心了,你不一定记得你有多少个组件监听了该事件,你可能会担心,那个组件到底挂载了没有。所以在复杂的应用中我们推荐使用vuex。

什么是vuex
  1. Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式,它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。vuex包含以下几个元素state(存放数据),action(动作,我把它理解为指令,用户派发action,可以看做是用户发出指令),mutation(改变,用来改变state里面的数据)一个完整的过程是视图(用户)派发 action,action提交修改,mutation负责修改state里的数据,state里的数据被修改了,对应的视图就被刷新,见下图。

    img_be68719a9e63469fb846d7e1dec92b81.png
  2. 每一个 Vuex 应用的核心就是 store(仓库)。“store”基本上就是一个容器,它包含着你的应用中大部分的状态 (state)。Vuex 和单纯的全局对象有相似的地方,都是把数据存到一个地方,然后取出来,或者进行数据的修改,但有以下两点不同:
    • Vuex 的状态存储是响应式的。当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新。
    • 你不能直接改变 store 中的状态。改变 store 中的状态的唯一途径就是显式地提交 (commit) mutation。这样使得我们可以方便地跟踪每一个状态的变化,从而让我们能够实现一些工具帮助我们更好地了解我们的应用。
一个vuex demo
  1. 打开上面的例子,在当前文件夹安装vuex
npm i vuex --save
  1. 新建组件Headline.vue,详细代码如下:
    vuex中提供了mapGetters、mapActions、mapMutations用来映射store里面的相应属性,方便我们直接使用我们事先定义的getters、actions和mutations。
  1. 在src文件夹下新建文件夹store,再新建文件index.js,在index.js里面添加以下内容
import Vue from 'vue'import Vuex from 'vuex'Vue.use(Vuex)export default new Vuex.Store({    // 存储数据    state: {        title: '根组件'    },    // 读取数据    getters: {        title: state => state.title    },    // 动作    actions: {        // 修改标题的action        'UPDATE_TITLE_ACTION'({ commit }, payload) {            commit('UPDATE_TITLE', payload);        }    },    // 变化;转变    mutations: {         // 修改state里的title        'UPDATE_TITLE'(state, payload) {            state.title = payload;        }    }})

当涉及的数据比较多的时候,getters、actions、mutations可以用单独的文件导出。

  1. 在main.js里添加下面两行代码
import store from './store'new Vue({  el: '#app',  // 新增代码  store,  router,  template: '
', components: { App }})
  1. 在Login组件里添加如下代码,当我们访问Login组件的时候,就可以看到页面上的组件名称发生了改变
beforeRouteEnter: (to, from, next) => {    next(vm => {      vm.$store.dispatch('UPDATE_TITLE_ACTION','登录组件')    })  },

同理,我们可以新建一个Register组件,做相关操作,当我们访问/register时改变组件的名称,Register组件代码如下,同时需要在router里面配置添加/register:

router/index.js里面的代码如下:

import Vue from 'vue'import Router from 'vue-router'import Index from '@/components/Index'import Login from '@/components/Login'import Register from '@/components/Register'Vue.use(Router)export default new Router({  routes: [    {      path: '/',      name: 'index',      component: Index    },    {      path: '/login',      name: 'login',      component: Login    },    {      path: '/register',      name: 'register',      component: Register    }  ]})

7.modules,当我们需要管理很多的状态时,会使用modules来进行分割,具体做法如下:

在store的文件夹新建modules文件夹,里面封装某个具体组件的state、getters、actions和mutations,一下新建一个Foot组件,设置关于该组件的状态信息,其他组件可以修改它的状态,从而改变Foot组件显示的内容(统计浏览过的路由的次数)。

  • 在components下新建组件Foot.vue,代码如下:
  • 在store/modules下新建foot.js,代码如下:
export default {    state: {        routeNum: '1'    },    getters: {        routeNum: state => state.routeNum    },    actions: {        'UPDATE_ROUTENUM_ACTION'({ commit }, payload) {            commit('UPDATE_ROUTENUM',payload);        }    },    mutations: {        'UPDATE_ROUTENUM'(state, payload) {            state.routeNum = state.routeNum + payload;        }     }}
  • 修改store/index.js,修改后代码如下:
import Vue from 'vue'import Vuex from 'vuex'import foot from './modules/footer'Vue.use(Vuex)// import getters from './getters'// import mutations from './mutations'// import actions from './actions';export default new Vuex.Store({    // 存储数据    state: {        title: '根组件'    },    // 读取数据    getters: {        title: state => state.title    },    actions: {        // 修改标题        'UPDATE_TITLE_ACTION'({ commit }, payload) {            commit('UPDATE_TITLE', payload);        }    },    mutations: {        'UPDATE_TITLE'(state, payload) {            state.title = payload;        }    },    modules: {        foot,    }})
  • 在其他组件去派发action来改变Foot组件的状态
    可以在每一个组件都加上这么一句
beforeRouteEnter: (to, from, next) => {    next(vm => {      vm.$store.dispatch("UPDATE_TITLE_ACTION", "登录组件");      vm.$store.dispatch("UPDATE_ROUTENUM_ACTION",1);    });  },

当然这个是比较笨的方法,更好的办法是,在main.js里面加上以下这段代码,那么当每一次变换路由的时候我们给底部加圆圈的小数字加1了。

router.beforeEach((to, from, next) => {  store.dispatch('UPDATE_ROUTENUM_ACTION',1);  next();})
在开始的event bus的例子里我们通过event bus改变组件的登录状态,在这里我们也可以使用vuex轻松实现,我们用state用loginState来保存登录的状态,当我们登录成功的时候,就修改loginState的值,具体实现过程如下。
  1. 在store/modules下新建文件login.js,具体代码如下:
import { mapActions } from "vuex";export default {    state: {        loginState: false    },    getters: {        loginState: state => state.loginState    },    actions: {        'UPDATE_LOGINSTATE_ACTION'({ commit }, payload) {            // UPDATE_LOGINSTATE为对应的mutation            commit('UPDATE_LOGINSTATE', payload);        }    },    mutations: {        'UPDATE_LOGINSTATE'(state, payload) {            state.loginState = payload;        }    }}
  1. 修改store/index.js,在modules里添加,修改后的代码如下:
import Vue from 'vue'import Vuex from 'vuex'import foot from './modules/footer'import login from './modules/login'Vue.use(Vuex)export default new Vuex.Store({    // 存储数据    state: {        title: '根组件'    },    // 读取数据    getters: {        title: state => state.title    },    actions: {        // 修改标题        'UPDATE_TITLE_ACTION'({ commit }, payload) {            commit('UPDATE_TITLE', payload);        }    },    mutations: {        'UPDATE_TITLE'(state, payload) {            state.title = payload;        }    },    modules: {        foot,        login    }})
  1. 修改login.vue,代码如下:
  1. 修改components下的index.vue,代码如下:
探讨问题:
  1. devtool
  2. vue路由的history模式
  3. 刷新后的store里面的数据清空,怎么处理

转载地址:http://xjukx.baihongyu.com/

你可能感兴趣的文章
Linux的netstat命令使用
查看>>
大快网站:如何选择正确的hadoop版本
查看>>
经过这5大阶段,你离Java程序员就不远了!
查看>>
IntelliJ IDEA 连接数据库详细过程
查看>>
PHP-X开发扩展
查看>>
android学习笔记——onSaveInstanceState的使用
查看>>
工作中如何做好技术积累
查看>>
Spring Transactional
查看>>
shell脚本实例
查看>>
我的友情链接
查看>>
Windows Phone 7 隔离存储空间资源管理器
查看>>
apache安装报错undefined reference ssl
查看>>
关于爱情只有一句忠告
查看>>
CentOS 7下安装部署Oracle11g图文教程
查看>>
F#初学笔记06
查看>>
实战:将企业域名解析委派给企业DNS服务器
查看>>
在Lync 2013环境部署Office Web Apps
查看>>
微软大会Ignite,你准备好了么?
查看>>
读书笔记-高标管事 低调管人
查看>>
Master带给世界的思考:是“失控”还是进化
查看>>