vuex入門,圖文+實例解析

我理解的概念:

vuex是爲vue提供了全局的狀態倉庫(store),就像一個狀態機,避免了父子、兄弟組件之前複雜的傳參。他維持了全局共用的數據的一致性。

核心概念秒懂:
1,state 共用的數據
2,getters 處理state後得到想要的數據
3,mutations 唯一可以修改state的函數
4,actions 只能顯式的調用mutations,可以異步、請求數據
5,moudles 把1、2、3、4包裝起來的當成一個模塊,可以有多個也可以沒有

不說廢話直接在實例裏面一一解釋:

項目結構:
在這裏插入圖片描述

安裝:

cnpm i vuex -S 

創建:

創建如圖的store
以下代碼都是moduleA代碼,

state.js

const state = {
  userInfo: {
    userName: '秋刀魚笛滋味',
    age: 28,
    job: '前端工程師'
  },
  firend: [],
  girlfirend: [
    {
      name: 'LuoSi',
      age: 20,
      nationality: '韓國'
    },
    {
      name: 'AnNi',
      age: 22,
      nationality: '俄羅斯'
    }
  ]
}
export default state;

state沒啥好解釋的就一個對象,放你要用的狀態碼

getters.js

const getters = {
  userJob: (state) => {
    return `${state.userInfo.job}`
  },
  girlfirendInfo: (state, getters) => {
    const girlfirend = state.girlfirend
    let info = girlfirend.map((item, index) => {
      return `${index + 1}號女友的名字是${item.name},年齡${item.age},來自${item.nationality}`
    }).join(',')
    return `一共有${girlfirend.length}個女友,${info},可怕的是他只是一名${getters.userJob}。`
  }
}
export default getters;

getters接受兩個參數,第一個是state,第二個是getters裏面其他的函數

mutation.js

import axios from 'axios';

const mutations = {
  ageAdd (state, payload) {
    payload = payload || 1
    state.userInfo.age += payload
  },
  addGirlFirend (state, payload) {
    state.girlfirend.push({ name: payload.name, age: payload.age, nationality: payload.nationality })
  },
  getFirend (state, payload) {
    state.firend = payload
  },
  mutfired (state) { //vuex嚴禁在mutations裏面進行異步操作,嚴格模式報錯,難於調試
    axios.get('/myServer').then(res => {
      if (res.status === 200) {
        state.firend = res.data.data.list
      }
    })
  }
}
export default mutations;

mutations接受兩個參數:state payload(調用時攜帶的參數),他是唯一可以修改state的地方,注意不可異步、不可調接口,嚴格模式會報錯
如圖:在這裏插入圖片描述

actions.js

import axios from 'axios';
const actions = {
  addGirlFirend ({ commit, state, getters }, payload) {
    commit('addGirlFirend', payload);
  },
  getFirends (ctx) { //ctx是store下當前module對象
    axios.get('/myServer').then(res => {
      if (res.status === 200) {
        ctx.commit('getFirend', res.data.data.list)
        //直接在actions裏面也可以修改state,但是不建議,創建store時用嚴格模式,會報錯,不符合vuex單向數據流的規範(只能在mutions裏面修改state)
        // ctx.state.firend = res.data.data.list
      }
    })
  }
}
export default actions;

actions接受一個當前module的上下文對象(常用有commit),用來commit 提交mutations,主要用來請求後端數據,可以異步

index.js

import state from './state';
import getters from './getters';
import mutations from './mutations.js';
import actions from './actions';

const moduleA = {
  state,
  getters,
  mutations,
  actions
}

export default moduleA;

把各個組件集合起來暴露出模塊

再來看看store的實例化:

store/index.js

import Vuex from 'vuex'
import Vue from 'vue'

import moduleA from './moduleA';
import moduleB from './moduleB';

Vue.use(Vuex)

let store = new Vuex.Store({
  //在嚴格模式下,無論何時發生了狀態變更且不是由 mutation 函數引起的,將會拋出錯誤。這能保證所有的狀態變更都能被調試工具跟蹤到。
  //*嚴格模式會深度監測狀態樹來檢測不合規的狀態變更——請確保在發佈環境下關閉嚴格模式,以避免性能損失。
  strict: process.env.NODE_ENV !== 'production',//自動在生產環境下關閉嚴格模式
  modules: {
    moduleA,
    moduleB
  }
})

export default store

注意:一定要用Vue.use一下vuex,最好使用嚴格模式!
當然store裏面還可以用命名空間和插件,一般項目用不上

掛載store

在項目主文件
main.js 實例化vue時,掛載

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

直接在實例裏面看怎麼快速使用store吧

helloWorld.vue

<template>
  <div>
    <Card style="width:550px">
      <div slot="title">
        <Icon type="ios-film-outline"></Icon>
        個人信息
      </div>
      <div>
        <p>姓名: {{userInfo.userName}}</p>
        <p>年齡: {{userInfo.age}}</p>

        {{girlfirendInfo}}
      </div>
    </Card>
    <hr style="margin:20px 0" />
    <Button type="success" @click="ageAdd()">增加了一歲</Button>
    <hr style="margin:20px 0" />
    <Button type="success" @click="addAge">增加了兩歲(commit)</Button>
    <hr style="margin:20px 0" />
    <Card style="width:550px">
      <div slot="title">
        <Icon type="ios-film-outline"></Icon>
        女友信息:
      </div>
      <div>
        名字:
        <Input v-model="girlInfo.name"></Input>
        年齡:</br>
        <Input-number :max="100" :min="1" v-model="girlInfo.age"></Input-number></br>
        國籍:
        <Input v-model="girlInfo.nationality"></Input>
      </div>
      <Button type="success" @click="addGirlFirend(girlInfo)">增加</Button>
      <Button type="success" @click="addGirlFirend1">增加(dispatch)</Button>
    </Card>
    <hr style="margin:20px 0" />

    <Card style="width:550px">
      <div slot="title">
        <Icon type="ios-film-outline"></Icon>
        朋友信息:
      </div>
      <div>
        <p v-for="item in firend" :key="item.userName">{{item.userName}}</p>
      </div>
      <Button type="info" @click="getFirends">獲取朋友</Button>
    </Card>
  </div>
</template>
<script>
import { mapState, mapGetters, mapMutations, mapActions } from 'vuex'
export default {
  data () {
    return {
      girlInfo: {
        name: '',
        age: 18,
        nationality: ''
      }
    }
  },
  computed: {
    ...mapGetters(['girlfirendInfo']),
    ...mapState({
      userInfo: state => state.moduleA.userInfo, //使用vuex的modules後一定要指明模塊
      firend: state => state.moduleA.firend
    })
  },
  methods: {
    ...mapActions(['addGirlFirend', 'getFirends']), //this.$store.dispatch('addGirlFirend',payload)
    ...mapMutations(['ageAdd']), //this.$store.commit('ageAdd',payload)
    // 上面兩個輔助函數方法的實質跟下面是一樣的,推薦 使用輔助函數
    addAge () {
      this.$store.commit('ageAdd', 2)
    },
    addGirlFirend1 () {
      this.$store.dispatch('addGirlFirend', this.girlInfo)
    }
  }
}

先看一下初始UI吧

在這裏插入圖片描述

簡單解釋一下

主要的4個模塊,有對應的四個輔助函數,用處是把狀態 和 操作映射到當前頁面

mapState 和 mapGetters,是狀態數據,放在計算屬性;
mapMutations和 mapActions 是操作函數, 顯然放在方法裏面;

注意帶的註釋;

直接看效果吧

調用mutations
在這裏插入圖片描述
調用actions
在這裏插入圖片描述
actions調接口
在這裏插入圖片描述
在這裏插入圖片描述

vuex的問題,解決方法點擊vuex刷新state就沒了

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章