摘要:在單頁面組件的開發(fā)中的和的都統(tǒng)稱為同一狀態(tài)管理,個人的理解是全局狀態(tài)管理更合適簡單的理解就是你在中定義了一個數(shù)據(jù)之后,你可以在所在項(xiàng)目中的任何一個組件里進(jìn)行獲取進(jìn)行修改,并且你的修改可以得到全局的響應(yīng)變更。
在SPA單頁面組件的開發(fā)中 Vue的vuex和React的Redux 都統(tǒng)稱為同一狀態(tài)管理,個人的理解是全局狀態(tài)管理更合適;簡單的理解就是你在state中定義了一個數(shù)據(jù)之后,你可以在所在項(xiàng)目中的任何一個組件里進(jìn)行獲取、進(jìn)行修改,并且你的修改可以得到全局的響應(yīng)變更。下面咱們一步一步地剖析下vuex的使用:
首先要安裝、使用 vuex
首先在 vue 2.0+ 你的vue-cli項(xiàng)目中安裝 vuex :
npm install vuex --save
然后 在src文件目錄下新建一個名為store的文件夾,為方便引入并在store文件夾里新建一個index.js,里面的內(nèi)容如下:
import Vue from "vue"; import Vuex from "vuex"; Vue.use(Vuex); const store = new Vuex.Store(); export default store;
接下來,在 main.js里面引入store,然后再全局注入一下,這樣一來就可以在任何一個組件里面使用this.$store了:
import store from "./store"http://引入store new Vue({ el: "#app", router, store,//使用store template: "", components: { App } })
說了上面的前奏之后,接下來就是納入正題了,就是開篇說的state的玩法?;氐絪tore文件的index.js里面,我們先聲明一個state變量,并賦值一個空對象給它,里面隨便定義兩個初始屬性值;然后再在實(shí)例化的Vuex.Store里面?zhèn)魅胍粋€空對象,并把剛聲明的變量state仍里面:
import Vue from "vue"; import Vuex from "vuex"; Vue.use(Vuex); const state={//要設(shè)置的全局訪問的state對象 showFooter: true, changableNum:0 //要設(shè)置的初始屬性值 }; const store = new Vuex.Store({ state }); export default store;
實(shí)際上做完上面的三個步驟后,你已經(jīng)可以用this.$store.state.showFooter或this.$store.state.changebleNum在任何一個組件里面獲取showfooter和changebleNum定義的值了,但這不是理想的獲取方式;vuex官方API提供了一個getters,和vue計(jì)算屬性computed一樣,來實(shí)時(shí)監(jiān)聽state值的變化(最新狀態(tài)),并把它也仍進(jìn)Vuex.Store里面,具體看下面代碼:
import Vue from "vue"; import Vuex from "vuex"; Vue.use(Vuex); const state={ //要設(shè)置的全局訪問的state對象 showFooter: true, changableNum:0 //要設(shè)置的初始屬性值 }; const getters = { //實(shí)時(shí)監(jiān)聽state值的變化(最新狀態(tài)) isShow(state) { //方法名隨意,主要是來承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //方法名隨意,主要是用來承載變化的changableNum的值 return state.changebleNum } }; const store = new Vuex.Store({ state, getters }); export default store;
光有定義的state的初始值,不改變它不是我們想要的需求,接下來要說的就是mutations了,mutattions也是一個對象,這個對象里面可以放改變state的初始值的方法,具體的用法就是給里面的方法傳入?yún)?shù)state或額外的參數(shù),然后利用vue的雙向數(shù)據(jù)驅(qū)動進(jìn)行值的改變,同樣的定義好之后也把這個mutations扔進(jìn)Vuex.Store里面,如下:
import Vue from "vue"; import Vuex from "vuex"; Vue.use(Vuex); const state={ //要設(shè)置的全局訪問的state對象 showFooter: true, changableNum:0 //要設(shè)置的初始屬性值 }; const getters = { //實(shí)時(shí)監(jiān)聽state值的變化(最新狀態(tài)) isShow(state) { //承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //承載變化的changebleNum的值 return state.changableNum } }; const mutations = { show(state) { //自定義改變state初始值的方法,這里面的參數(shù)除了state之外還可以再傳額外的參數(shù)(變量或?qū)ο?; state.showFooter = true; }, hide(state) { //同上 state.showFooter = false; }, newNum(state,sum){ //同上,這里面的參數(shù)除了state之外還傳了需要增加的值sum state.changableNum+=sum; } }; const store = new Vuex.Store({ state, getters, mutations }); export default store;
這時(shí)候你完全可以用 this.$store.commit("show") 或 this.$store.commit("hide") 以及 this.$store.commit("newNum",6) 在別的組件里面進(jìn)行改變showfooter和changebleNum的值了,但這不是理想的改變值的方式;因?yàn)樵?Vuex 中,mutations里面的方法 都是同步事務(wù),意思就是說:比如這里的一個this.$store.commit("newNum",sum)方法,兩個組件里用執(zhí)行得到的值,每次都是一樣的,這樣肯定不是理想的需求
好在vuex官方API還提供了一個actions,這個actions也是個對象變量,最大的作用就是里面的Action方法 可以包含任意異步操作,這里面的方法是用來異步觸發(fā)mutations里面的方法,actions里面自定義的函數(shù)接收一個context參數(shù)和要變化的形參,context與store實(shí)例具有相同的方法和屬性,所以它可以執(zhí)行context.commit(" "),然后也不要忘了把它也扔進(jìn)Vuex.Store里面:
import Vue from "vue"; import Vuex from "vuex"; Vue.use(Vuex); const state={ //要設(shè)置的全局訪問的state對象 showFooter: true, changableNum:0 //要設(shè)置的初始屬性值 }; const getters = { //實(shí)時(shí)監(jiān)聽state值的變化(最新狀態(tài)) isShow(state) { //承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //承載變化的changebleNum的值 return state.changableNum } }; const mutations = { show(state) { //自定義改變state初始值的方法,這里面的參數(shù)除了state之外還可以再傳額外的參數(shù)(變量或?qū)ο?; state.showFooter = true; }, hide(state) { //同上 state.showFooter = false; }, newNum(state,sum){ //同上,這里面的參數(shù)除了state之外還傳了需要增加的值sum state.changableNum+=sum; } }; const actions = { hideFooter(context) { //自定義觸發(fā)mutations里函數(shù)的方法,context與store 實(shí)例具有相同方法和屬性 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;
而在外部組件里進(jìn)行全局執(zhí)行actions里面方法的時(shí)候,你只需要用執(zhí)行
this.$store.dispatch("hideFooter")
或this.$store.dispatch("showFooter")
以及this.$store.dispatch("getNewNum",6) //6要變化的實(shí)參
這樣就可以全局改變改變showfooter或changebleNum的值了,如下面的組件中,需求是跳轉(zhuǎn)組件頁面后,根據(jù)當(dāng)前所在的路由頁面進(jìn)行隱藏或顯示頁面底部的tabs選項(xiàng)卡
}else{ this.$store.dispatch("hideFooter") } } } }
至此就可以做到一呼百應(yīng)的全局響應(yīng)狀態(tài)改變了!
modules 模塊化 以及 組件中引入 mapGetters、mapActions 和 mapStates的使用
因?yàn)樵诖蠖鄶?shù)的項(xiàng)目中,我們對于全局狀態(tài)的管理并不僅僅一種情況的需求,有時(shí)有多方面的需求,比如寫一個商城項(xiàng)目,你所用到的全局state可能是關(guān)于購物車這一塊兒的也有可能是關(guān)于商品價(jià)格這一塊兒的;像這樣的情況我們就要考慮使用vuex中的 modules 模塊化了,具體怎么使用modules呢?咱們繼續(xù)一步一步的走:
首先,在store文件夾下面新建一個modules文件夾,然后在modules文件里面建立需要管理狀態(tài)的js文件,既然要把不同部分的狀態(tài)分開管理,那就要把它們給分成獨(dú)立的狀態(tài)文件了,如下圖:
而對應(yīng)的store文件夾下面的index.js 里面的內(nèi)容就直接改寫成:
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 } });
相應(yīng)的js,其中的 namespaced:true 表示當(dāng)你需要在別的文件里面使用( mapGetters、mapActions 接下來會說 )時(shí),里面的方法需要注明來自哪一個模塊的方法:
//collection.js const state={ collects:[], //初始化一個colects數(shù)組 }; const getters={ renderCollects(state){ //承載變化的collects return state.collects; } }; const mutations={ pushCollects(state,items){ //如何變化collects,插入items state.collects.push(items) } }; const actions={ invokePushItems(context,item){ //觸發(fā)mutations里面的pushCollects ,傳入數(shù)據(jù)形參item 對應(yīng)到items context.commit("pushCollects",item); } }; export default { namespaced:true,//用于在全局引用此文件里的方法時(shí)標(biāo)識這一個的文件名 state, getters, mutations, actions }
//footerStatus.js const state={ //要設(shè)置的全局訪問的state對象 showFooter: true, changableNum:0 //要設(shè)置的初始屬性值 }; const getters = { //實(shí)時(shí)監(jiān)聽state值的變化(最新狀態(tài)) isShow(state) { //承載變化的showFooter的值 return state.showFooter }, getChangedNum(){ //承載變化的changebleNum的值 return state.changableNum } }; const mutations = { show(state) { //自定義改變state初始值的方法,這里面的參數(shù)除了state之外還可以再傳額外的參數(shù)(變量或?qū)ο?; state.showFooter = true; }, hide(state) { //同上 state.showFooter = false; }, newNum(state,sum){ //同上,這里面的參數(shù)除了state之外還傳了需要增加的值sum state.changableNum+=sum; } }; const actions = { hideFooter(context) { //自定義觸發(fā)mutations里函數(shù)的方法,context與store 實(shí)例具有相同方法和屬性 context.commit("hide"); }, showFooter(context) { //同上注釋 context.commit("show"); }, getNewNum(context,num){ //同上注釋,num為要變化的形參 context.commit("newNum",num) } }; export default { namespaced: true, //用于在全局引用此文里的方法時(shí)標(biāo)識這一個的文件名 state, getters, mutations, actions }
這樣一改就有了關(guān)于兩個模塊的state管理文件了 footerStatus.js和collection.js,現(xiàn)在你要運(yùn)行當(dāng)前的代碼話,項(xiàng)目會報(bào)錯!因?yàn)槲覀儼焉厦娴拇a模塊化分開了,引用的地方還沒有改。接下來咱們一起來看看 mapState,mapGetters,mapActions的使用,首先 在需要用的 組件里面先導(dǎo)入 import {mapState,mapGetters,mapActions} from "vuex";咱們先修正一下隱藏或顯示頁面底部的tabs選項(xiàng)卡(就是上面舉的臨時(shí)例子)的組件代碼
現(xiàn)在項(xiàng)目代碼應(yīng)該就不會報(bào)錯了,好,最后咱們再來看一下mapActions的用法,實(shí)際上上面的this.$store.dispatch("footerStatus/showFooter")已經(jīng)算是一種執(zhí)行相應(yīng)模塊的action里的方法了,但有時(shí)會牽扯的事件的觸發(fā)及傳值,那就會有下面的mapActions用法了,還記得上面的另一個模塊collection.js嗎?來看一下里面的actions中的方法結(jié)構(gòu):
const state={ collects:[], //初始化一個colects數(shù)組 }; const getters={ renderCollects(state){ //承載變化的collects return state.collects; } }; const mutations={ pushCollects(state,items){ //如何變化collects,插入items state.collects.push(items) } }; const actions={ invokePushItems(context,item){ //觸發(fā)mutations里面的pushCollects ,傳入數(shù)據(jù)形參item 對應(yīng)到items context.commit("pushCollects",item); } };
需要傳值來實(shí)時(shí)變動state.collects里的數(shù)據(jù),那肯定要在執(zhí)行它的地方進(jìn)行傳值了,所以下面用到它的地方我們用了個@click來執(zhí)行這個invokePushItems方法了,并且傳入相應(yīng)的對象數(shù)據(jù)item,如下:
全國改性料通訊錄 加入收藏列
這樣一來,在這個組件里面操作的 collecttion.js 中的state的數(shù)據(jù),在其他的任何的一個組件里面都會得到相應(yīng)的更新變化了,獲取狀態(tài)的頁面代碼如下:
{{val.price}}
至此,vuex中的常用的一些知識點(diǎn)使用算是簡單的分享完了,當(dāng)然了,相信這些只是一些皮毛!只能說是給予剛接觸vuex的初學(xué)者一個參考與了解吧!有哪里不明白的或不對的,留言下,咱們可以一起討論、共同學(xué)習(xí)!
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/113583.html
摘要:在單頁面組件的開發(fā)中的和的都統(tǒng)稱為同一狀態(tài)管理,個人的理解是全局狀態(tài)管理更合適簡單的理解就是你在中定義了一個數(shù)據(jù)之后,你可以在所在項(xiàng)目中的任何一個組件里進(jìn)行獲取進(jìn)行修改,并且你的修改可以得到全局的響應(yīng)變更。 在SPA單頁面組件的開發(fā)中 Vue的vuex和React的Redux 都統(tǒng)稱為同一狀態(tài)管理,個人的理解是全局狀態(tài)管理更合適;簡單的理解就是你在state中定義了一個數(shù)據(jù)之后,你可以...
摘要:把打包的目錄修改成生產(chǎn)環(huán)境需要的目錄。是域名的配置只要統(tǒng)一配置一項(xiàng)即可,方便。旨在增強(qiáng)團(tuán)隊(duì)開發(fā)協(xié)作提高代碼質(zhì)量和打造開發(fā)基石的編碼規(guī)范,以下規(guī)范是團(tuán)隊(duì)基本約定的內(nèi)容,必須嚴(yán)格遵循。 Vue作為前端三大框架之一,其已經(jīng)悄然成為主流,學(xué)會用vue相關(guān)技術(shù)來開發(fā)項(xiàng)目會相當(dāng)輕松。 對于還沒學(xué)習(xí)或者還沒用過vue的初學(xué)者,基礎(chǔ)知識這里不作詳解,推薦先去相關(guān)官網(wǎng),學(xué)習(xí)一下vue相關(guān)的基礎(chǔ)知識。 a...
摘要:我相信很多小白跟我一樣一接觸到時(shí)候也是一頭霧水吧,我從實(shí)戰(zhàn)一書中學(xué)習(xí)到的知識再加上自己的理解做了一個簡單明了的截圖地址歡迎給我到時(shí)候我也會把高級方法的理解結(jié)合一起更新 我相信很多小白跟我一樣一接觸到vuex時(shí)候也是一頭霧水吧,我從一書中學(xué)習(xí)到的知識再加上自己的理解做了一個簡單明了的demodemo 截圖showImg(https://segmentfault.com/img/bVZiS...
摘要:踩過的坑安卓低版本兼容性處理阻止表單中,默認(rèn)事件,出現(xiàn)刷新問題。踩過的坑: axios 安卓低版本兼容性處理 阻止表單中,button默認(rèn)事件,出現(xiàn)刷新問題。 設(shè)置滾動條的位置 vue 數(shù)據(jù)和對象數(shù)據(jù)變化 dom結(jié)構(gòu)不變 android低版本 白屏問題 你是不是用了Object.assign ?? ? 相關(guān)鏈接 學(xué)習(xí)文檔: vue官網(wǎng):https://cn.vuejs.org/v2/guid...
閱讀 3079·2021-11-24 10:34
閱讀 3336·2021-11-22 13:53
閱讀 2639·2021-11-22 12:03
閱讀 3607·2021-09-26 09:47
閱讀 3014·2021-09-23 11:21
閱讀 4814·2021-09-22 15:08
閱讀 3302·2021-07-23 10:59
閱讀 1269·2019-08-29 18:31