vuex學習及使用

什麼是vuex?vue

在SPA單頁面組件的開發中vuex稱爲狀態管理;簡單的理解就是你在state中定義了一個數據以後,你能夠在所在項目中的任何一個組件裏進行獲取、進行修改,而且你的修改能夠獲得全局的響應變動。vuex

在vue中使用vuex,首先要安裝:npm

npm install vuex --save

而後再src文件目錄下新建一個名爲store的文件夾,爲方便引入並在store文件夾裏新建一個index.js,裏面的內容以下:segmentfault

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'//引入store

new Vue({
  el:'#app',
  router,
  store,//使用store
  template:'<App/>',
  components:{App}
})

說了上面的前奏以後,接下來就是歸入正題了,就是開篇說的state的玩法。回到store文件的index.js裏面,咱們先聲明一個state變量,並賦值一個空對象給它,裏面隨便定義兩個初始屬性值;而後再實例化的Vuex.Store裏面傳入一個空對象,並把剛聲明的變量state扔裏面:app

import Vue from 'vue';
import Vuex from 'vuex';
Vue.use(Vuex);
const state={//要設置的全局訪問的state對象
  showFooter:true,
  changableNum:0
  //要設置的初始屬性值
};
const store=new Vuex.Store({
  state
});

export default store;

實際上作完上面的三個步驟後,你已經能夠用this.$store.state.showFooter或this.$store.state.changebleNum在任何一個組件裏面獲取showFooter和changebleNum定義的值了,但這不是理想的獲取方式;vuex官方API提供了一個getters,和vue計算屬性computed同樣,來實時監聽state值得變化(最新狀態),並把它扔進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.changebleNum
  }
};
const store=new Vuex.Store({
  state,
  getters
});
export default store;

光有定義的state的初始值,不改變它不是咱們想要的需求,接下來要說的就是mutations了,mutattions也是一個對象,這個對象裏面能夠放改變state的初始值的方法,具體的用法就是給裏面的方法傳入參數state或額外的參數,而後利用vue的雙向數據驅動進行值得改變,一樣的定義好以後把這個mutations扔進Vuex.Store裏面,以下:ide

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(){
    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和changableNum的值了,但這不是理想的改變值得方式;由於在Vuex中,mutations裏面的方法都是同步事務,意思就是說:好比這裏的一個this.$store.commit('newNum',sum)方法,兩個組件裏用執行獲得的值,每次都是同樣的,這樣確定不是理想的需求模塊化

好在vuex官方API還提供了一個actions,這個actions也是個對象變量,最大的做用就是裏面的Actions方法能夠包含任意異步操做,這裏面的方法是用力啊異步觸發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(){//承載幻化的changableNum的值
    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或changableNum的值了,以下面的組件中,需求是跳轉組件頁面後,根據當前所在的路由頁面進行隱藏或顯示頁面底部的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來獲取當前所顯示的頁面,從而控制該顯示或隱藏footerVar組件
        this.$store.dispatch('showFooter')//利用派發全局state.showFooter的值來控制
      }else{
        this.$store.dispatch('hideFooter')
      }
    }
  }
}
</script>

 至此就能夠作到一呼百應的全局響應狀態改變了!

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,如今你要運行當前的代碼話,項目會報錯!由於咱們把上面的代碼模塊化分開了,引用的地方尚未改。接下來我們一塊兒來看看 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>

完結!

學習來源:https://segmentfault.com/a/1190000015782272

相關文章
相關標籤/搜索