什麼是vuex,官網的描述是:Vuex 是一個專爲 Vue.js 應用程序開發的狀態管理模式。
狀態管理模式包含3個部分:vue
一、state,驅動應用的數據源;vuex
二、view,以聲明方式將state映射到視圖;npm
三、actions,響應在view上的用戶輸入致使的狀態變化。segmentfault
接下來講說它們是如何工做的。數組
首先說說如何在項目中使用vuexapp
首先安裝vuex:npm install vuex --save
異步
而後 在src文件目錄下新建一個名爲store的文件夾,並在store文件夾裏新建一個index.js,裏面的內容以下ide
import Vue from 'vue'; import Vuex from 'vuex'; Vue.use(Vuex); //使用vuex const store = new Vuex.Store(); //定義store export default store; //暴露store接口
接着,在項目的 main.js裏面引入store,而後再全局注入一下模塊化
import store from './store'//引入store new Vue({ el: '#app', router, store,//使用store,即全局注入 template: '<App/>', components: { App } })
這樣就能夠在任何一個組件裏面使用this.$store了。函數
搭建好vuex後,開始進入正題,如何使用vuex。
回到store文件的index.js裏面,聲明一個state變量,並賦值一個對象給它,對象裏面隨便定義兩個初始屬性值;而後再在實例化的Vuex.Store裏面傳入一個空對象,並把剛聲明的變量state仍裏面:
import Vue from 'vue'; import Vuex from 'vuex'; Vue.use(Vuex); const state={//要設置的全局訪問的state對象 showFooter: true, changableNum:0 //要設置的初始屬性值 }; const store = new Vuex.Store({ state //把上面聲明的state對象扔到實例化的Vuex.Store裏 }); export default store;
至此,你就能夠在任何一個組件裏,經過this.$store.state.showFooter或this.$store.state.changebleNum獲取showfooter和changebleNum定義的值了。
but這不是理想的獲取方式。
vuex官方API提供了一個getters,來實時監聽state值的變化(最新狀態),照樣在store文件的index.js裏面修改,具體看下面:
import Vue from 'vue'; import Vuex from 'vuex'; Vue.use(Vuex); const state={ //要設置的全局訪問的state對象 showFooter: true, changableNum:0 //要設置的初始屬性值 }; const getters = { //實時監聽state值的變化(最新狀態) isShow(state) { //方法名隨意,主要是來承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //方法名隨意,主要是用來承載變化的changableNum的值 return state.changebleNum } }; const store = new Vuex.Store({ state, getters }); export default store;
getters 只用來監聽(獲取)state的值,並不能用來改變state的值。改變state的值須要用mutations。照樣在store文件的index.js裏面修改,具體代碼以下:
import Vue from 'vue'; import Vuex from 'vuex'; Vue.use(Vuex); const state={ //要設置的全局訪問的state對象 showFooter: true, changableNum:0 //要設置的初始屬性值 }; const getters = { //實時監聽state值的變化(最新狀態) isShow(state) { //承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //承載變化的changebleNum的值 return state.changableNum } }; const mutations = { show(state) { //自定義改變state初始值的方法,這裏面的參數除了state以外還能夠再傳額外的參數(變量或對象); state.showFooter = true; }, hide(state) { //同上 state.showFooter = false; }, newNum(state,sum){ //同上,這裏面的參數除了state以外還傳了須要增長的值sum state.changableNum+=sum; } }; const store = new Vuex.Store({ state, getters, mutations }); export default store;
mutattions也是一個對象,這個對象裏面能夠建立改變state的初始值的方法,方法裏面的參數用state或者額外的參數,
定義好以後一樣把mutations扔進Vuex.Store裏面。
這時候就徹底能夠用 this.$store.commit('show') 或 this.$store.commit('hide') 以及 this.$store.commit('newNum',6)
在別的組件裏面進行改變showfooter和changebleNum的值了,
但這不是理想的改變值的方式。
由於在 Vuex 中,mutations裏面的方法都是同步事務,意思就是說:
好比這裏的一個this.$store.commit('newNum',sum)方法,每一個組件裏用執行獲得的值,每次都是同樣的,這樣確定不是理想的需求。
vuex官方API還提供了一個actions,這個actions也是個對象變量,最大的做用就是裏面的Action方法 能夠包含任意異步操做,
這裏面的方法是用來異步觸發mutations裏面的方法,actions裏面自定義的函數接收一個context參數和要變化的形參,
context與store實例具備相同的方法和屬性,因此它能夠執行context.commit(' '),而後把actions也扔進Vuex.Store裏面:
照樣在store文件的index.js裏面修改,具體代碼以下:
import Vue from 'vue'; import Vuex from 'vuex'; Vue.use(Vuex); const state={ //要設置的全局訪問的state對象 showFooter: true, changableNum:0 //要設置的初始屬性值 }; const getters = { //實時監聽state值的變化(最新狀態) isShow(state) { //承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //承載變化的changebleNum的值 return state.changableNum } }; const mutations = { show(state) { //自定義改變state初始值的方法,這裏面的參數除了state以外還能夠再傳額外的參數(變量或對象); state.showFooter = true; }, hide(state) { //同上 state.showFooter = false; }, newNum(state,sum){ //同上,這裏面的參數除了state以外還傳了須要增長的值sum state.changableNum+=sum; } }; const actions = { hideFooter(context) { //自定義觸發mutations裏函數的方法,context與store 實例具備相同方法和屬性 context.commit('hide');//commit裏面的參數即爲mutations對象裏面的方法 }, showFooter(context) { //同上註釋 context.commit('show');//同上註釋 }, getNewNum(context,num){ //同上註釋,num爲要變化的形參 context.commit('newNum',num)//同上註釋 } }; const store = new Vuex.Store({ state, getters, mutations, actions }); export default store;
而在外部組件裏進行全局執行actions裏面方法的時候,你只須要用執行this.$store.dispatch('hideFooter')或this.$store.dispatch('showFooter')以及this.$store.dispatch('getNewNum',6) //6要變化的實參。
這樣就能夠全局改變改變showfooter或changebleNum的值了。
以下面的組件中,需求是跳轉組件頁面後,根據當前所在的路由頁面進行隱藏或顯示頁面底部的tabs選項卡
<template> <div id="app"> <router-view/> <FooterBar v-if="isShow" /> </div> </template> <script> import FooterBar from '@/components/common/FooterBar' import config from './config/index' export default { name: 'App', components:{ FooterBar:FooterBar }, data(){ return { } }, computed:{ isShow(){ return this.$store.getters.isShow; } }, watch:{ $route(to,from){ //跳轉組件頁面後,監聽路由參數中對應的當前頁面以及上一個頁面 console.log(to) if(to.name=='book'||to.name=='my'){ // to.name來獲取當前所顯示的頁面,從而控制該顯示或隱藏footerBar組件 this.$store.dispatch('showFooter') // 利用派發全局state.showFooter的值來控制 }else{
this.$store.dispatch('hideFooter') }
}
</script>
至此就能夠作到一呼百應的全局響應狀態改變了!
modules 模塊化 以及 組件中引入 mapGetters、mapActions 和 mapStates的使用
在實際項目中,不少時候須要模塊管理,好比寫一個商城項目,你
所用到的全局state多是關於購物車這一起的也有多是關於商品價格這一起的;像這樣的狀況咱們就要考慮使用vuex
中的 modules 模塊化了,具體怎麼使用modules呢?我們繼續一步一步的走:
首先,在store文件夾下面新建一個modules文件夾,在modules文件夾裏面創建須要管理狀態的js文件,不一樣的狀態須要用不一樣的js文件來管理,以下圖:
在store文件夾下面的index.js 裏面引入這兩個文件,內容改寫成:
import Vue from 'vue'; import Vuex from 'vuex'; import footerStatus from './modules/footerStatus' import collection from './modules/collection' Vue.use(Vuex); export default new Vuex.Store({ modules:{ footerStatus, collection } });
相應的js文件以下:
//collection.js const state={ collects:[], //初始化一個colects數組 }; const getters={ renderCollects(state){ //承載變化的collects return state.collects; } }; const mutations={ pushCollects(state,items){ //如何變化collects,插入items state.collects.push(items) } }; const actions={ invokePushItems(context,item){ //觸發mutations裏面的pushCollects ,傳入數據形參item 對應到items context.commit('pushCollects',item); } }; export default { namespaced:true,//用於在全局引用此文件裏的方法時標識這一個的文件名 state, getters, mutations, actions }
//footerStatus.js const state={ //要設置的全局訪問的state對象 showFooter: true, changableNum:0 //要設置的初始屬性值 }; const getters = { //實時監聽state值的變化(最新狀態) isShow(state) { //承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //承載變化的changebleNum的值 return state.changableNum } }; const mutations = { show(state) { //自定義改變state初始值的方法,這裏面的參數除了state以外還能夠再傳額外的參數(變量或對象); state.showFooter = true; }, hide(state) { //同上 state.showFooter = false; }, newNum(state,sum){ //同上,這裏面的參數除了state以外還傳了須要增長的值sum state.changableNum+=sum; } }; const actions = { hideFooter(context) { //自定義觸發mutations裏函數的方法,context與store 實例具備相同方法和屬性 context.commit('hide'); }, showFooter(context) { //同上註釋 context.commit('show'); }, getNewNum(context,num){ //同上註釋,num爲要變化的形參 context.commit('newNum',num) } }; export default { namespaced: true, //用於在全局引用此文裏的方法時標識這一個的文件名 state, getters, mutations, actions }
其中的 namespaced:true 表示當你須要在別的文件裏面使用( mapGetters、mapActions 接下來會說 )時,裏面的方法須要註明來自哪個模塊的方法。
就這樣,collection.js和footerStatus.js兩個文件就寫好了。若是如今運行當前的代碼話,項目會報錯!由於上面的代碼模塊化分開了,引用的地方尚未改。接下來一塊兒來看看
mapState,mapGetters,mapActions的使用,首先 在須要用的 組件裏面先導入 import {mapState,mapGetters,mapActions} from 'vuex';
我們先修正一下隱藏或顯示頁面底部的tabs選項卡(就是上面舉的臨時例子)的組件代碼
<template> <div id="app"> <router-view/> <FooterBar v-if="isShow" /> </div> </template> <script> import {mapState,mapGetters,mapActions} from 'vuex'; //先要引入 import FooterBar from '@/components/common/FooterBar' import config from './config/index' export default { name: 'App', components:{ FooterBar:FooterBar }, data(){ return { } }, computed:{ ...mapState({ //這裏的...是超引用,ES6的語法,意思是state裏有多少屬性值我能夠在這裏放多少屬性值 isShow:state=>state.footerStatus.showFooter //注意這些與上面的區別就是state.footerStatus。showFooter是指footerStatus.js裏state的showFooter }), //你也能夠用下面的mapGetters來獲取isShow的值,貌似下面的更簡潔 /*...mapGetters('footerStatus',{ //footerStatus指的是modules文件夾下的footerStatus.js模塊 isShow:'isShow' //第一個isShow是我自定義的只要對應template裏v-if="isShow"就行, //第二個isShow是對應的footerStatus.js裏的getters裏的isShow })*/ }, watch:{ $route(to,from){ if(to.name=='book'||to.name=='my'){ this.$store.dispatch('footerStatus/showFooter') //這裏改成'footerStatus/showFooter',意思是指footerStatus.js裏actions裏的showFooter方法 }else{ this.$store.dispatch('footerStatus/hideFooter') //同上註釋 } } } } </script>
如今項目代碼應該就不會報錯了,接下來看一下mapActions的用法。
實際上上面的this.$store.dispatch('footerStatus/showFooter')已經算是一種執行相應模塊的action裏的方法了,但有時會牽扯的事件的觸發及傳值,那就會有下面的mapActions用法了,還記得上面的另外一個模塊collection.js嗎?來看一下里面的actions中的方法結構:
const state={ collects:[], //初始化一個colects數組 }; const getters={ renderCollects(state){ //承載變化的collects return state.collects; } }; const mutations={ pushCollects(state,items){ //如何變化collects,插入items state.collects.push(items) } }; const actions={ invokePushItems(context,item){ //觸發mutations裏面的pushCollects ,傳入數據形參item 對應到items context.commit('pushCollects',item); } };
須要傳值來實時變更state.collects裏的數據,那確定要在執行它的地方進行傳值了,因此下面用到它的地方咱們用了個@click來執行這個invokePushItems方法了,而且傳入相應的對象數據item,以下
<template> <div > <section class="joinState"> <div class="joinStateHead"> <span class="h3">全國改性料通信錄</span> <span class="joinStatus" @click="invokePushItems(item)">加入收藏列</span> </div> </section> </div> </template> <script> import { mapActions } from 'vuex' export default { components:{ conditionFilter }, name: 'bookDetail', data () { return { msg: '', item:{ id:'01', productName: '蘋果', price:'1.6元/斤' } } }, mounted() { this.$store.dispatch('footerStatus/hideFooter') }, methods:{ ...mapActions('collection',[ //collection是指modules文件夾下的collection.js 'invokePushItems' //collection.js文件中的actions裏的方法,在上面的@click中執行並傳入實參 ]) } } </script>
這樣一來,在這個組件裏面操做的 collecttion.js 中的state的數據,在其餘的任何的一個組件裏面都會獲得相應的更新變化了,獲取狀態的頁面代碼以下:
<template> </div> <div> <ul> <li v-for="(val,index) in arrList" :key="index"> <h5>{{val.productName}}</h5> <p>{{val.price}}</p> </li> </ul> </div> </template> <script> import {mapState,mapGetters,mapActions} from 'vuex'; export default { name: 'book', data() { return { } }, computed:{ // ...mapState({ //用mapState來獲取collection.js裏面的state的屬性值 // arrList:state=>state.collection.collects // }), ...mapGetters('collection',{ //用mapGetters來獲取collection.js裏面的getters arrList:'renderCollects' }) } } </script>
至此,vuex中的經常使用的一些知識點使用算是基本瞭解了。
參考文獻:https://segmentfault.com/a/1190000015782272