nuxt踩坑之Vuex狀態(tài)樹的模塊方式使用詳解
初次看到這個模塊方式,感覺很是新奇,之前的vuex狀態(tài)樹使用方法用的也有些膩了,就想來實踐一發(fā)新的東西
廢話不多說,直接進入正題
Vuex狀態(tài)樹-模塊方式官方文檔解讀
狀態(tài)樹還可以拆分成為模塊,store 目錄下的每個 .js 文件會被轉(zhuǎn)換成為狀態(tài)樹指定命名的子模塊
- 這句話啊,看了半天,我都沒繞出來。之前一直用的是store目錄下文件為:index.js、state.js、mutations.js、actions.js。后三個是index.js的子模塊,你說這每個js文件都是一個模塊?懵逼一分鐘
- 繼續(xù)往下:使用狀態(tài)樹模塊化的方式,store/index.js 不需要返回 Vuex.Store 實例,而應(yīng)該直接將 state、mutations 和 actions 暴露出來。 到這里我雖然還懵逼著,但是轉(zhuǎn)念一想,nuxt還有這操作,都不用我們自己辛辛苦苦寫export default store = () => new Vuex.Store({}) 了,倒也真的省事兒呢
- 還是繼續(xù)看例子吧,官方給的這個,看了一遍沒看懂(笨小孩的世界真滴難),你這index.js不是Vuex默認的store文件么,再來一todos.js,同樣暴露出去的對象,不應(yīng)該是index.js同級的么
- 重點來了,看不會不要緊,照貓畫虎我還是會滴~
照貓畫虎
// store/index.js export const state = () => ({ num: 0 }) export const mutations = { increment (state) { state.num ++ }, decrement (state) { state.num -- } } // store/plus.js export const state = () => ({ plusNum: 1 }) export const mutations = { plus (state) { state.plusNum ++ } } // store/minus.js export const state = () => ({ minusNum: 10 }) export const mutations = { minus (state) { state.minusNum -- } } // pages/store.vue <template> <section class="container"> <table> <tr> <td colspan=4>vuex狀態(tài)樹使用</td> </tr> <tr> <td>頁內(nèi)數(shù)據(jù)</td> <td>index.js</td> <td>plus.js</td> <td>minus.js</td> </tr> <tr> <td>{{ count }}</td> <td>{{ $store.state.num }}</td> <td>{{ $store.state.plus.plusNum }}</td> <td>{{ $store.state.minus.minusNum }}</td> </tr> <tr> <td></td> <td></td> <td></td> <td></td> </tr> </table> </section> </template>
跑一下,?。箦e了,我說同學們啊,我寫的真的沒有錯?。?!
好吧,報錯內(nèi)容:[Vue warn]: The client-side rendered virtual DOM tree is not matching server-rendered content. This is likely caused by incorrect HTML markup, for example nesting block-level elements inside <p>, or missing <tbody>. Bailing hydration and performing full client-side render.
意思是我客戶端和vue SSR生成的DOM不一樣,客戶端不也是SSR生成的,這是個問題,有知道的大佬,希望可以告訴我。
不過這個問題我倒是解決了,雖然不知道問什么-_-!,把那一大堆的tr標簽都放到tbody里面就OK了
跑起來,沒問題,按示例的寫法,這樣成功拿到了state的數(shù)據(jù)
接著試試mutation的方法
<tr class="mutation-fun"> <td @click="count ++">count ++</td> <td @click="$store.commit('increment')">increment</td> <td @click="$store.commit('plus')">plus</td> <td @click="$store.commit('minus')">minus</td> </tr>
報錯:[vuex] unknown mutation type: plus
修改下:
<tr class="mutation-fun"> <td @click="count ++">count ++</td> <td @click="$store.commit('increment')">increment</td> <td @click="$store.commit('plus/plus')">plus/plus</td> <td @click="$store.commit('minus/minus')">minus/minus</td> </tr>
搞定~
自己先小結(jié)下這個模塊怎么用的吧
nuxt很貼心的幫我們省去了返回Vuex實例的代碼,我們可以不用去寫了
只有store文件夾下的index.js是一級的vuex狀態(tài),其他的js文件都是二級的狀態(tài)樹。(能不能有三級的我不知道,不過感覺沒必要,哈哈哈?。。?br />
每個狀態(tài)樹文件都可以包含state,mutation,action
使用二級狀態(tài)樹的state用: $store.state.文件名.變量名
使用二級狀態(tài)樹的mutation用: $store.commit(‘文件名/變量名')
使用二級狀態(tài)樹的action用: $store.dispatch(‘文件名/變量名')
官方示例沒有提到的
二級狀態(tài)樹能調(diào)用一級狀態(tài)樹的state和mutation以及action嗎?
我們來給plus.js增加一個修改index.js中的state的方法plusIndex
export const state = () => ({ plusNum: 1 }) export const mutations = { plus (state) { state.plusNum ++ }, plusIndex (state) { state.num ++ console.log('點擊遞增index的num') } }
去試著調(diào)用了一下,文本打印出來了,也沒有報錯,但是完全獲取不到index中的num哇~~
這次聰明點兒,我直接把state打印出來:
export const state = () => ({ plusNum: 1 }) export const mutations = { plus (state) { state.plusNum ++ }, plusIndex (state) { console.log('state: ', state) state.num ++ } }
跑一下試試: 結(jié)果有點兒讓人想哭:state: {num: NaN, plusNum: 1}。再看看頁面上index.js對應(yīng)的num依然是0,這就說明我們這里的num不是index.js里的num,而是下面state ‘.'(點)出來的num,并且沒有給初始值就執(zhí)行了一次 ++ ,所以值為NaN
到這里,我已經(jīng)大概猜測到了這里的彎彎繞是怎么回事兒了 –> 模塊兒的作用域是本文件內(nèi)。但編程不是靠猜的,要用實踐證明的才是對的,下面在深究這個問題
反過來,一級狀態(tài)樹能調(diào)用二級狀態(tài)樹的state和mutation以及action嗎?
這里的情況跟上面一樣,既然我們的這個store是模塊方式生成的,那就要遵循模塊化的規(guī)范,變量作用域只能在文件內(nèi)
通過模塊化編程規(guī)范來獲取一級狀態(tài)樹的state
先把index.js引過來,因為不放心引用過來的還是不是原來那個index.js,所以把它打印出來看看
const indexVuex = require('./index.js') console.log('indexVuex: ', indexVuex)
這下不就炸了嘛,我引過來的居然是一個空的vuex狀態(tài)樹?。?!內(nèi)容長這樣子
{ mutation: {}, modules: { plus: { mutation: {}, namespaced: true }, minus: { mutation: {}, namespaced: true } }, namespaced: true }
好吧,我這個用模塊化的方式來使用其他文件內(nèi)的變量的想法也是泡湯了
最后的讓步,我在組件中調(diào)用狀態(tài)樹時再去用二級狀態(tài)樹的方法修改一級狀態(tài)樹的state,先試試:
// plus.js export const state = () => ({ plusNum: 1 }) export const mutations = { plus (state) { state.plusNum ++ }, plusIndex (state, meio) { meio ++ console.log('meio: ', meio) } } // 調(diào)用: <td @click="$store.commit('plus/plusIndex', $store.state.num)">遞增index的num</td>
點擊后,依然沒能修改index.js中的num,查看打印結(jié)果中一直都是1,也就是說我們傳進去的是基本數(shù)據(jù)類型的0~~
那修改一下,我們傳進去一個引用數(shù)據(jù)類型的看看可以不
// plus.js export const state = () => ({ plusNum: 1 }) export const mutations = { plus (state) { state.plusNum ++ }, plusIndex (state, meio) { meio.num ++ console.log('meio: ', meio.num) } } // 調(diào)用: <td @click="$store.commit('plus/plusIndex', $store.state)">遞增index的num</td>
功夫不負有心人,終于是成功了,雖然這樣很麻煩,但也起到了效果
這里經(jīng)驗證:反過來一級狀態(tài)樹調(diào)用二級狀態(tài)樹也是一樣的,代碼就不貼了
小結(jié)一下
- vuex狀態(tài)樹的模塊寫法,不同的文件之間不能直接引用
- vuex狀態(tài)樹使用模塊加載引用,得到的是一組命名空間,而不是最終生成的Vuex模塊文件
- 要通過b文件修改其他文件的state需要在store外部去修改
最后,小小的STAR下
情境(situation):粗略的看完一遍NUXT的官方文檔,回頭把不太清楚的東西再整理一下,這兒就卡在Vuex狀態(tài)樹的的模塊寫法上
任務(wù)(task):通過各種方法弄明白NUXT的Vuex狀態(tài)樹的模塊寫法是怎么實現(xiàn)的
行動(action):第一步,參照官方示例先讓自己的狀態(tài)樹能運行起來;第二步,與普通寫法的狀態(tài)樹寫法比較并舉一反三,看看模塊寫法和普通寫法的狀態(tài)樹在使用上有什么區(qū)別和需要注意的
結(jié)果(result):第一,學會了vuex狀態(tài)樹的模塊寫法和用法;第二,通過探索store目錄下的不同文件之間是否可以互相引用以及模塊引用來熟悉模塊方式的寫法和普通方式的區(qū)別,也找到了模塊寫法中修改其他模塊的state的方法;第三,模塊寫法的優(yōu)點:寫法簡單,結(jié)構(gòu)清晰明了,符合現(xiàn)在模塊化開發(fā);同文件(模塊)內(nèi)部類似普通寫法,可按業(yè)務(wù)模塊劃分Vuex模塊,提高開發(fā)效率;還有一點兒,我覺得這樣寫省事兒,也好看-,-| 媽媽再也不用擔心我所有的Vuex都放同一個文件中,要用時還得找半天了~~
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
vue3+vite引入插件unplugin-auto-import的方法
這篇文章主要介紹了vue3+vite引入插件unplugin-auto-import的相關(guān)知識,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值?,需要的朋友可以參考下2022-10-10基于vue3+TypeScript實現(xiàn)一個簡易的Calendar組件
最近在學習 react,在學習到使用 react 開發(fā) Calendar 組件的時候,突然聯(lián)想到使用 vue 進行 Calendar 功能的實現(xiàn),因為目前使用的技術(shù)棧是 vue,剛好可以加深下對 vue3 和 ts 的使用印象,所以本文給大家介紹了基于vue3+TypeScript實現(xiàn)一個簡易的Calendar組件2024-05-05vue-virtual-scroll-list虛擬組件實現(xiàn)思路詳解
這篇文章主要給大家介紹了關(guān)于vue-virtual-scroll-list虛擬組件實現(xiàn)思路的相關(guān)資料,文中通過實例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2023-02-02使用vue引入maptalks地圖及聚合效果的實現(xiàn)
這篇文章主要介紹了使用vue引入maptalks地圖及聚合效果的實現(xiàn),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-08-08vue的異步數(shù)據(jù)更新機制與$nextTick用法解讀
這篇文章主要介紹了vue的異步數(shù)據(jù)更新機制與$nextTick用法解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03React/vue開發(fā)報錯TypeError:this.getOptions?is?not?a?function
這篇文章主要給大家介紹了關(guān)于React/vue開發(fā)報錯TypeError:this.getOptions?is?not?a?function的解決方法,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2023-07-07Vue2中Element?DatePicker組件設(shè)置默認日期及控制日期范圍
后臺項目想使用時間選擇器選擇一段時間進行數(shù)據(jù)篩選,所以下面這篇文章主要給大家介紹了關(guān)于Vue2中Element?DatePicker組件設(shè)置默認日期及控制日期范圍的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2022-11-11