1. mutations
mutations也是一個對象,這個對象裏面可以放改變state的初始值的方法,具體的用法就是給裏面的方法傳入參數state或額外的參數,然後利用vue的雙向數據驅動進行值的改變,同樣的定義好之後也把這個mutations扔進Vuex.Store裏面,如下:
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;
這時候你完全可以用 this.$store.commit('show') 或 this.$store.commit('hide') 以及 this.$store.commit('newNum',6) 在別的組件裏面進行改變showfooter和changebleNum的值了。
2. actions
但這不是理想的改變值的方式;因爲在 Vuex 中,mutations裏面的方法 都是同步事務,意思就是說:比如這裏的一個this.$store.commit('newNum',sum)方法,兩個組件裏用執行得到的值,每次都是一樣的,這樣肯定不是理想的需求。
好在vuex官方API還提供了一個actions,這個actions也是個對象變量,最大的作用就是裏面的Action方法 可以包含任意異步操作,這裏面的方法是用來異步觸發mutations裏面的方法,actions裏面自定義的函數接收一個context參數和要變化的形參,context與store實例具有相同的方法和屬性,所以它可以執行context.commit(' '),然後也不要忘了把它也扔進Vuex.Store裏面:
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');
},
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的值了
例子:如下面的組件中,需求是跳轉組件頁面後,根據當前所在的路由頁面進行隱藏或顯示頁面底部的FooterBar組件
<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>
3. modules模塊化
modules 模塊化 以及 組件中引入 mapGetters、mapActions 和 mapStates的使用
因爲在大多數的項目中,我們對於全局狀態的管理並不僅僅一種情況的需求,有時有多方面的需求,比如寫一個商城項目,你所用到的全局state可能是關於購物車這一塊兒的也有可能是關於商品價格這一塊兒的;像這樣的情況我們就要考慮使用vuex中的 modules 模塊化了,具體怎麼使用modules呢?咱們繼續一步一步的走:
首先,在store文件夾下面新建一個modules文件夾,然後在modules文件裏面建立需要管理狀態的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,其中的 namespaced:true 表示當你需要在別的文件裏面使用( mapGetters、mapActions 接下來會說 )時,裏面的方法需要註明來自哪一個模塊的方法:
//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
}
這樣一改就有了關於兩個模塊的state管理文件了 footerStatus.js和collection.js,現在你要運行當前的代碼話,項目會報錯!因爲我們把上面的代碼模塊化分開了,引用的地方還沒有改。
4.mapState,mapGetters,mapActions的使用
接下來咱們一起來看看 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中的常用的一些知識點使用算是簡單的分享完了,當然了,相信這些只是一些皮毛!只能說是給予剛接觸vuex的初學者一個參考與瞭解吧!有哪裏不明白的或不對的,留言下,咱們可以一起討論、共同學習!