做者:小土豆biubiubiujavascript
博客園:www.cnblogs.com/HouJiao/css
掘金:juejin.im/user/58c61b…html
簡書:www.jianshu.com/u/cb1c3884e…前端
微信公衆號:土豆媽的碎碎念(掃碼關注,一塊兒吸貓,一塊兒聽故事,一塊兒學習前端技術) vue
歡迎你們掃描微信二維碼進入羣聊討論(若二維碼失效可添加微信JEmbrace拉你進羣) ![]()
碼字不易,點贊鼓勵喲~java
上一篇文章 [手把手式教程,適合新手入門Vuex]-Vuex入門實踐(上),咱們一共實踐了Vuex
的這些內容。web
在state
中定義共享屬性,在組件中可以使用$store.state.屬性名
訪問共享屬性。vuex
在mutations
中定義修改共享數據
的方法,在組件中可以使用$store.commit('方法名')
同步修改共享屬性。瀏覽器
在actions
中定義異步修改共享數據
的方法,在組件中可以使用$store.dispatch('方法名')
異步修改共享屬性。微信
在getters
中定義共享數據的計算屬性
,在組件中可以使用$store.getters.計算屬性名
訪問共享數據的計算屬性。
那本篇文章我將帶你們完成Vuex
的多模塊實踐。
同時本篇文章也是Vuex
系列的第二篇,第一篇請 點擊此處 閱讀。
前面的文章中,咱們在Vuex
的實例方法Store
上分別建立了state
、mutations
、actions
和getters
這幾個配置選項。
能夠思考一下,當咱們的應用程序越發複雜時,組件之間須要共享的數據也在持續增長,那麼state
、mutations
、actions
和getters
配置項裏面的代碼會越發龐大。
所以爲解決此問題,Vuex
支持每一個模塊定義本身的state
、mutations
、actions
和getters
。
如今咱們來實踐一下多個module
。
首先咱們在E:\MyStudy\test\VueDemo\src\vuex
目錄下新建兩個文件:moduleA.js
和moduleB.js
。
接着分別編輯這兩個文件的state
、mutations
、actions
和getters
配置項。
// E:\MyStudy\test\VueDemo\src\vuex\moduleA.js
const moduleA = {
state:{
counter: 100
},
mutations: {
//遞增
increase(state) {
state.counter++
},
//遞減
decrement(state) {
state.counter--
}
},
actions: {
increaseAction(context) {
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
    context.commit('increase');
},3000)
},
decrementAction(context){
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
     context.commit('decrement');
},3000)
}
},
getters: {
doubleCounter(state) {
return state.counter*state.counter
}
}
}
export default moduleA
複製代碼
// E:\MyStudy\test\VueDemo\src\vuex\moduleB.js
const moduleB = {
state:{
counter: 5
},
mutations: {
//遞增
increase(state) {
state.counter++
},
//遞減
decrement(state) {
state.counter--
}
},
actions: {
increaseAction(context) {
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
context.commit('increase');
},3000)
},
decrementAction(context){
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
     context.commit('decrement');
},3000)
}
},
getters: {
doubleCounter(state){
return state.counter*state.counter
}
}
}
export default moduleB
複製代碼
能夠看到每一個module
定義本身的state
、mutations
、actions
和getters
和直接寫在store.js
中的語法相差無幾,只是單個module
無需將這些選項掛載到vuex
實例的Store
方法上。
最後就是須要在store.js
中編寫該模塊的state
、mutations
、actions
和getters
,而且配置這兩個module
。
// E:\MyStudy\test\VueDemo\src\vuex\store.js
import Vue from 'vue'
import Vuex from 'vuex'
import moduleA from './moduleA'
import moduleB from './moduleB'
Vue.use(Vuex)
export default new Vuex.Store({
state: {
counter: 1000
},
mutations: {
//遞增
increase(state) {
state.counter++
},
//遞減
decrement(state) {
state.counter--
}
},
actions: {
increaseAction(context) {
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
     context.commit('increase');
},3000)
},
decrementAction(context){
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
     context.commit('decrement');
},3000)
}
},
getters: {
doubleCounter(state) {
return state.counter*state.counter
}
},
modules: {
a: moduleA,
b: moduleB
}
})
複製代碼
store.js
中對多個module
的配置語法也比較簡單,即在modules
中以字典的形式模塊名:模塊引用
寫入便可。其他配置state
、mutations
、actions
和getters
的方法和前面的如出一轍。
Vuex
多個module
的配置完成,如今咱們分別按順序訪問store
模塊、a
模塊和b
模塊的共享數據
、同步/異步修改共享數據
以及計算屬性訪問
。
多個Module
這種狀況下,state
中的共享數據被綁定在模塊上(模塊內部的state
是局部的,只屬於模塊自己),所以咱們須要使用$store.state.模塊名稱
去訪問不一樣模塊的共享數據。而對於store.js
這個根模塊
配置的state
數據,直接使用$store.state
訪問便可。
那麼總結一下:
訪問store根模塊counter的邏輯:$store.state.counter
訪問a模塊counter的邏輯爲:$store.state.a.counter
訪問b模塊的counter的邏輯爲:$store.state.b.counter
複製代碼
如今咱們分別在App.vue
組件中訪問store.js
這個根模塊的counter
和a
模塊的counter
數據,在Index.vue
組件中訪問b
模塊的counter
數據。
在那個組件中訪問那個模塊的
state
無所謂,也能夠捨棄Index.vue
,將全部代碼都寫在App.vue
組件中。我在實踐時分開寫的目的只是想體現多個組件之間能夠共享Vuex
中定義的數據。
<!-- E:\MyStudy\test\VueDemo\src\App.vue -->
<template>
<div id="app">
<img src="./assets/logo.png">
<!-- 獲取共享數據 -->
<h1>這裏是App組件</h1>
<h3> App組件獲取共享數據 </h3>
<h3>訪問根模塊counter——$store.state.counter : {{ $store.state.counter }} </h3>
<h3>訪問a模塊counter——$store.state.a.counter : {{ $store.state.a.counter }} </h3>
<hr/>
<Index></Index>
</div>
</template>
<script> import Index from './components/Index' export default { name: 'App', components: { Index } } </script>
<style> #app { font-family: 'Avenir', Helvetica, Arial, sans-serif; -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale; text-align: center; color: #2c3e50; margin-top: 60px; } </style>
複製代碼
<!-- E:\MyStudy\test\VueDemo\src\components\Index.vue -->
<template>
<div>
<h1>這裏是Index.vue組件</h1>
<h3>Index組件獲取共享數據 </h3>
<h3>訪問b模塊counter——$store.state.b.counter :{{ $store.state.b.counter }}</h3>
</div>
</template>
<script> export default { name: 'Index' } </script>
複製代碼
最後咱們啓動項目查看一下結果。
能夠看到,咱們已經成功的訪問到了不一樣模塊的counter
數據。
如今,咱們須要在App.vue
組件中同步修改store
根模塊、a
模塊的數據,在Index.vue
組件中同步修改b
模塊的數據。
前面咱們說了state
綁定在模塊上,而mutations
它並無和模塊綁定。那麼根據 [手把手式教程,適合新手入門Vuex]-Vuex入門實踐(上) 這一篇文章中的總結,得知觸發counter
遞增的方法爲$store.commit(‘increase’)
。
因爲store
根模塊、a
模塊、b
模塊中使counter
遞增和遞減的mutations
方法名都是如出一轍
的,那麼當咱們之間使用$store.commit(‘increase’)
去觸發counter
遞增會出現什麼樣的結果呢?咱們來試一下。
首先在store
根模塊、a
模塊、b
模塊中的mutations
increase
中添加打印console.log
的打印信息,其他代碼沿用前面的保存不變。
// E:\MyStudy\test\VueDemo\src\vuex\store.js
//遞增
increase(state) {
console.log("store-increase")
state.counter++
}
複製代碼
// E:\MyStudy\test\VueDemo\src\vuex\moduleA.js
//遞增
increase(state) {
console.log("moduleA-increase")
state.counter++
},
複製代碼
// E:\MyStudy\test\VueDemo\src\vuex\moduleB.js
//遞增
increase(state) {
console.log("moduleB-increase")
state.counter++
},
複製代碼
接着編寫App.vue
和Index.vue
的代碼。
<!-- E:\MyStudy\test\VueDemo\src\App.vue -->
<template>
<div id="app">
<img src="./assets/logo.png">
<!-- 獲取共享數據 -->
<h1>這裏是App組件</h1>
<h3> App組件獲取共享數據 </h3>
<h3>訪問根模塊counter——$store.state.counter : {{ $store.state.counter }} </h3>
<h3>訪問a模塊counter——$store.state.a.counter : {{ $store.state.a.counter }} </h3>
<h3>同步修改根模塊counter:<button @click="$store.commit('increase')">同步修改根模塊counter</button></h3>
<h3>同步修改a模塊counter:<button @click="$store.commit('increase')">同步修改a模塊counter</button></h3>
<hr/>
<Index></Index>
</div>
</template>
<script> import Index from './components/Index' export default { name: 'App', components: { Index } } </script>
<style> #app { font-family: 'Avenir', Helvetica, Arial, sans-serif; -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale; text-align: center; color: #2c3e50; margin-top: 60px; } </style>
複製代碼
<!-- E:\MyStudy\test\VueDemo\src\components\Index.vue -->
<template>
<div>
<h1>這裏是Index.vue組件</h1>
<h3>Index組件獲取共享數據 </h3>
<h3>訪問b模塊counter——$store.state.b.counter :{{ $store.state.b.counter }}</h3>
<h3>同步修改b模塊counter:<button @click="$store.commit('increase')">同步修改b模塊counter</button></h3>
</div>
</template>
<script> export default { name: 'Index' } </script>
複製代碼
最後咱們來看下效果。
那麼現象就是:執行一次$store.commit('increase')
會依次觸發store
根模塊、a
模塊和b
模塊mutations
中increase
的執行。因此看到當點擊button
時這三個模塊的counter
都會加1
。
關於觸發
increase
的順序呢,先是根模塊
,而後是根據根模塊
中modules
的配置順序決定的。
這個操做呢只是我本身的一個小嚐試,真正多模塊的mutations
也不會這樣去使用。那麼想避免上面問題的出現,咱們能夠避免多個模塊中的mutations
方法的重名。
咱們來實踐一下,store
模塊中mutations
的increase
名稱不作修改,只修改a
、b
這兩個模塊。
// E:\MyStudy\test\VueDemo\src\vuex\moduleA.js
mutations: {
//遞增
increaseA(state) {
console.log("moduleA-increase")
state.counter++
}
},
複製代碼
// E:\MyStudy\test\VueDemo\src\vuex\moduleB.js
mutations: {
//遞增
increaseB(state) {
console.log("moduleB-increase")
state.counter++
}
},
複製代碼
而後對應的須要修改App.vue
和Index.vue
中觸發mutations
的代碼(只貼出修改部分代碼)。
<!-- E:\MyStudy\test\VueDemo\src\App.vue -->
<h3>同步修改a模塊counter:<button @click="$store.commit('increaseA')">同步修改a模塊counter</button></h3>
複製代碼
<!-- E:\MyStudy\test\VueDemo\src\components\Index.vue -->
<h3>同步修改b模塊counter:<button @click="$store.commit('increaseB')">同步修改b模塊counter</button></h3>
複製代碼
如今在看下效果。
能夠看到不一樣的按鈕觸發了不一樣的模塊的mutations
。
那麼還有一個問題,當module
特別多時,其實不可避免真的會有重名的mutations
,那麼此時Vuex
的命名空間
就須要上場了。
命名空間
也很簡單,就是在模塊中添加一個配置namespaced:true
。
如今咱們來實踐一下分別給模塊a
和模塊b
添加命名空間的配置,同時在把mutations
中遞增的方法名稱統一改回increase
。
// E:\MyStudy\test\VueDemo\src\vuex\moduleA.js
// 省略部分代碼
const moduleA = {
// 模塊A命令空間配置
namespaced: true,
state:{
counter: 100
},
mutations: {
//遞增
increase(state) {
console.log("moduleA-increase")
state.counter++
},
//遞減
decrement(state) {
state.counter--
}
}
}
export default moduleA
複製代碼
// E:\MyStudy\test\VueDemo\src\vuex\moduleB.js
// 省略部分代碼
const moduleB = {
// 模塊B命令空間配置
namespaced: true,
state:{
counter: 5
},
mutations: {
//遞增
increase(state) {
console.log("moduleB-increase")
state.counter++
},
//遞減
decrement(state) {
state.counter--
}
}
}
export default moduleB
複製代碼
當有了命令空間之後,觸發mutations
的方法也就有了變化: $store.commit('模塊名/方法')
。
<!-- E:\MyStudy\test\VueDemo\src\App.vue -->
<h3>同步修改根模塊counter:<button @click="$store.commit('increase')">同步修改根模塊counter</button></h3>
<h3>同步修改a模塊counter:<button @click="$store.commit('a/increase')">同步修改a模塊counter</button></h3>
複製代碼
<!-- E:\MyStudy\test\VueDemo\src\components\Index.vue -->
<h3>同步修改b模塊counter:<button @click="$store.commit('b/increase')">同步修改b模塊counter</button></h3>
複製代碼
看下結果。
能夠看到命名空間
的效果和前面修改mutations名稱不重複
是一樣的效果。
異步修改共享數據的邏輯和前面同步修改的相同。
咱們在actions
中定義了異步同名的遞增和遞減方法,執行一次$store.dispatch('increaseAction')
,會依次觸發執行store
、a
模塊和b
模塊的actions
。一樣咱們能夠選擇讓不一樣模塊的actions
方法名稱不重複,也可使用命名空間去解決。
這裏咱們只演示命名空間
的方式去觸發不一樣module
的actions
。
// E:\MyStudy\test\VueDemo\src\vuex\moduleA.js
const moduleA = {
// 模塊A命令空間配置
namespaced: true,
state:{
counter: 100
},
mutations: {
//遞增
increase(state) {
console.log("moduleA-increase")
state.counter++
},
//遞減
decrement(state) {
state.counter--
}
},
actions: {
increaseAction(context) {
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
context.commit('increase');
},3000)
},
decrementAction(context){
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
context.commit('decrement');
},3000)
}
},
}
export default moduleA
複製代碼
// E:\MyStudy\test\VueDemo\src\vuex\moduleB.js
const moduleB = {
// 模塊B命令空間配置
namespaced: true,
state:{
counter: 5
},
mutations: {
//遞增
increase(state) {
console.log("moduleB-increase")
state.counter++
},
//遞減
decrement(state) {
state.counter--
}
},
actions: {
increaseAction(context) {
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
context.commit('increase');
},3000)
},
decrementAction(context){
setTimeout(function(){
//action經過提交mutation改變共享數據狀態
context.commit('decrement');
},3000)
}
},
}
複製代碼
store.js保持不變
接着是組件中觸發actions
的代碼。
<!-- E:\MyStudy\test\VueDemo\src\App.vue -->
<h3>異步修改根模塊counter:<button @click="$store.dispatch('increaseAction')">異步修改根模塊counter</button></h3>
<h3>異步修改a模塊counter:<button @click="$store.dispatch('a/increaseAction')">異步修改a模塊counter</button></h3>
複製代碼
<!-- E:\MyStudy\test\VueDemo\src\components\Index.vue -->
<h3>異步修改b模塊counter:<button @click="$store.dispatch('b/increaseAction')">異步修改b模塊counter</button></h3>
複製代碼
如今咱們看下結果。
能夠看到配置命令空間後,已經成功的觸發了不一樣模塊同名的actions
。
最後一部分就是多module
的getters
訪問了。
首先這個須要說明的是getters
也沒有和模塊進行綁定,在咱們沒有給a
和b
模塊添加命名空間namespaced:true
的配置前。由於多個模塊的getters
存在重名屬性,所以控制檯能夠看到一個錯誤信息。
後面咱們在moduleA.js
和moduleB.js
中添加了命令空間
的配置後該錯誤就不會在出現。我本身也測試了一下,一樣能夠像前面那樣,保證getters
中的屬性不重名,直接使用$store.getters.屬性名
去訪問不一樣模塊的getters
。
下面來分別實踐一下兩種訪問getters
的方式。
store.js
中getters
的屬性名不作修改,依然是doubleCounter
;將moduleA.js
中getters
的屬性名改成doubleCounterA
;將moduleB.js
中getters
的屬性名改成doubleCounterB
。
// E:\MyStudy\test\VueDemo\src\vuex\moduleA.js
// 省略部分代碼
getters: {
doubleCounterA(state) {
return state.counter*state.counter
}
}
複製代碼
// E:\MyStudy\test\VueDemo\src\vuex\moduleB.js
getters: {
doubleCounterB(state) {
return state.counter*state.counter
}
}
複製代碼
接着就是在App.vue
和Index.vue
中訪問store
模塊、a
模塊和b
模塊的計算屬性。
<!-- E:\MyStudy\test\VueDemo\src\App.vue -->
<h3>訪問根模塊getters——$store.getters.doubleCounter : {{ $store.getters.doubleCounter }} </h3>
<h3>訪問a模塊getters——$store.getters.doubleCounterA : {{ $store.getters.doubleCounterA }} </h3>
複製代碼
<!-- E:\MyStudy\test\VueDemo\src\components\Index.vue -->
<h3>訪問b模塊getters——$store.getters.doubleCounterB : {{ $store.getters.doubleCounterB }} </h3>
複製代碼
瀏覽器查看結果。
能夠看到已經成功的訪問到不一樣模塊的getters
。
那麼最後咱們在嘗試將a
、b
兩個模塊的getters
屬性名稱改回doubleCounter
,使用命名空間
的方式去訪問。
這裏不貼
moduleA.js
和moudleB.js
的代碼了,直接修改App.vue
和Index.vue
中訪問getters
的代碼
<!-- E:\MyStudy\test\VueDemo\src\App.vue -->
<h3>訪問根模塊getters——$store.getters.doubleCounter : {{ $store.getters.doubleCounter }} </h3>
<h3>訪問a模塊getters——$store.getters['a/doubleCounter'] : {{ $store.getters['a/doubleCounter'] }} </h3>
複製代碼
<!-- E:\MyStudy\test\VueDemo\src\components\Index.vue -->
<h3>訪問b模塊getters——$store.getters[''b/doubleCounter'] : {{ $store.getters['b/doubleCounter'] }} </h3>
複製代碼
瀏覽器查看結果。
能夠看到命名空間訪問成功。
前面訪問
getters
的邏輯代碼爲$store.getters.doubleCounter
。所以在嘗試使用命名空間訪問getters
時個人代碼爲$store.getters.a.doubleCounter
。可是發現這種方法會報錯,所以靈機一動把代碼換成了$store.getters['a/doubleCounter']
,最後訪問成功。
到此本篇文章要總結的內容就完成了。篇幅比較長,可是也很簡單。
Vuex
的前一篇文章和本篇文章,都是經過$store
對象對state
、mutations
、actions
和getters
進行訪問和觸發的,所以下一篇文章將會介紹另一種比較經常使用的訪問觸發方式。
做者:小土豆biubiubiu
簡書:www.jianshu.com/u/cb1c3884e…
微信公衆號:土豆媽的碎碎念(掃碼關注,一塊兒吸貓,一塊兒聽故事,一塊兒學習前端技術)
歡迎你們掃描微信二維碼進入羣聊討論(若二維碼失效可添加微信JEmbrace拉你進羣) ![]()
碼字不易,點贊鼓勵喲~