Heim  >  Artikel  >  Web-Frontend  >  Leicht zu verstehen! Ausführliche Erläuterung der staatlichen Lagerverwaltung von VUEX

Leicht zu verstehen! Ausführliche Erläuterung der staatlichen Lagerverwaltung von VUEX

藏色散人
藏色散人nach vorne
2022-08-10 14:25:311749Durchsuche

Vuex ist ein Zustandsverwaltungsmuster, das speziell für Vue.js-Anwendungen entwickelt wurde. Es verwendet einen zentralen Speicher, um den Status aller Komponenten der Anwendung zu verwalten, und stellt mithilfe entsprechender Regeln sicher, dass sich der Status auf vorhersehbare Weise ändert. Um es einfach auszudrücken: Wenn die Anwendung auf den gemeinsamen Status mehrerer Komponenten stößt, verwenden Sie vuex.

Vue-Kernkonzept:

Status: Gemeinsamer Status, die Grunddaten von Vuex, die zum Speichern von Variablen verwendet werden. Dies entspricht den Daten in den Komponentendaten, wird jedoch jetzt zu einer globalen Variablen.

Getter: Abgeleiteter Zustand basierend auf dem Zustand, äquivalent zu den berechneten Eigenschaften in der Komponente.
  • Mutation: Eine Methode zum Ändern des gemeinsamen Status des Speichers in vuex. Ändern Sie den Status, indem Sie eine Mutation einreichen und die Daten synchron verarbeiten. Sie wird normalerweise für Aktionen verwendet, um asynchrone Daten zu erhalten und an die Mutation übermittelte Daten zu erhalten. und betreiben Sie die Daten im Zustand synchron in der Mutation.
  • Aktion: Unterstützt asynchrone Vorgänge, die verwendet werden können, um die Daten in der Anforderung asynchron abzurufen und die erhaltenen Daten synchron an die Mutation zu senden, um eine asynchrone Ajax-Anforderung für Daten zu implementieren, und die Mutation synchronisiert ihre Daten mit dem Status.
  • Module: Modulares Vuex. Um das spätere Projektmanagement zu erleichtern, kann jedes Modul seinen eigenen Status, seine eigene Mutation, seine eigene Aktion und seine eigenen Getter haben, wodurch die Struktur sehr klar und einfach zu verwalten ist.
  • Was sind die Vor- und Nachteile?
  • Der Hauptvorteil besteht darin, dass Daten und Methoden global geteilt werden können. Praktisch für eine einheitliche Verwaltung
  • Nachteil: Statusvariablen werden nach der Seitenaktualisierung wiederhergestellt und gelöscht und nicht dauerhaft wie Cookies gespeichert

Wie kann das Problem des Verlusts von Vuex-Statusdaten nach der Seitenaktualisierung gelöst werden?
Lassen Sie uns zunächst darüber sprechen, warum es verloren gegangen ist.
Da die Daten im Store im laufenden Speicher gespeichert sind, lädt die Seite beim Aktualisieren der Seite die Vue-Instanz neu und die Daten im Store werden neu zugewiesen

Wie kann man das vermeiden?
Tatsächlich hängt es hauptsächlich vom Nutzungsszenario ab. Wenn Sie bestimmte Daten dauerhaft speichern möchten, können Sie auch Cookies oder localStorage verwenden. Zum Beispiel einige Anmeldeinformationen usw.
Nachdem Sie beispielsweise die Anmeldeinformationen angefordert haben, können Sie diese zunächst im LocalStorage speichern, den Variablenwert im Status an den Wert im SessionStorage binden und beim Ändern der Mutationen gleichzeitig den Status und den LocalStorage ändern. Auf der letzten Seite werden Variablen in Vuex direkt verwendet. [Verwandte Empfehlungen:

vue.js Video-Tutorial

]

Öffnen Sie offiziell die Installation von

vuex
Öffnen Sie das Terminal und geben Sie die Befehlszeile npm install vuex --save ein, um den vuex

vuex Application Core Management Warehouse Build Store herunterzuladen


Erstellen Sie hier einen neuen Store-Ordner, erstellen Sie ein JS mit dem Namen index.js.

Führen Sie im Index Status, Mutationen, Aktionen und Getter in den Store ein und machen Sie das Store-Objekt verfügbar.

Das Folgende ist der Code von index.js

/*
  vuex最核心的管理对象 store
*/import Vue from 'vue';import Vuex from 'vuex';
 // 分别引入这四个文件  这四个文件的内容和用法在下面分别讲到import state from './state';import mutations from './mutations';import actions from './actions';import getters from './getters';
 //声明使用插件Vue.use(Vuex)//new 一个Vuex的对象,将state,mutation,action,getters配置到vuex的store中,方便管理数据export default new Vuex.Store({  
  state,
  mutations,
  actions,
  getters,})

Mounten Sie den Store in der Vue-Instanz

main.js

import store from './store'// ..........new Vue({
  el: '#app',
  router,
  store,   // ***
  render: h => h(App)})

Statusverwaltungsdaten


Normalerweise versetzen wir die gemeinsam genutzten Daten, die verwaltet werden müssen, in den zu erstellenden Status Es sieht aus wie eine globale Variable und führt die Statusdaten in die erforderlichen Komponenten ein.

const state = {
  userId: '',
  token: '',
  name: '',
  avatar: '',
  introduction: '',
  roles: [],
  tenantId: 1,
  userInfo: null};

Mutationen übermitteln synchron Daten

Mutationen werden verwendet, um die Zustandslogik im Staat zu ändern und die Zustandsdaten im Staat synchron zu ändern.

Was Sie wissen müssen, ist, dass das Statusobjekt nur durch Mutationen in Vuex geändert werden kann.

Sie können den Status ändern, indem Sie die aus Aktionen erhaltenen Daten abrufen, oder Sie können Methoden direkt im Mutationsmodul definieren, um die Statusdaten zu ändern.

const mutations = {
  SET_TOKEN: (state, token) => {
    state.token = token;
  },
  SET_USERID: (state, userId) => {
    state.userId = userId;
  },
  SET_NAME: (state, name) => {
    state.name = name;
  },
  SET_ROLES: (state, roles) => {
    state.roles = roles;
  },
  SET_TENANTID: (state, roles) => {
    state.tenantId = roles;
  },
  SET_USER_INFO: (state, userInfo) => {
    state.userInfo = userInfo;
  }};

Über die Mutationen und das Aktionsmodul unten können Sie auch sehen, dass Commit zum Aufrufen des Mutationsmoduls verwendet wird.
Der Code zum Aufrufen seines Mutationsmoduls in der Komponente lautet:

this.$store.commit('SET_TOKEN', token_data)

Der asynchrone Vorgang von Aktionen


Aktionen ähneln seinen Mutationen, können jedoch asynchrone Vorgänge ausführen,

und die durch den asynchronen Vorgang erhaltenen Daten an Mutationen senden. Damit Mutationen den Status ändern Die Statusdaten werden häufig verwendet, um die Daten in der Ajax-Anforderung abzurufen (da sie asynchron sind) und die erhaltenen Daten an Mutationen zu übergeben, um den Status der Statusdaten zu aktualisieren. Der Unterschied zwischen

und Mutationen ist:

Aktion übermittelt eine Mutation, anstatt den Zustand direkt zu ändern.

Action kann jede asynchrone Operation enthalten.
  1. Zum Beispiel
  2. /* 下面就是通过actions执行异步Ajax请求,
    得到数据后,
    通过commit的方法调用mutations 从而更新数据
    例如:  commit('SET_TOKEN', data.uuid);
    */const actions = {
      login({ commit }, userInfo) {  // 用户登录
        const params = userInfo;
        params.userName = userInfo.userName.trim()
        return new Promise((resolve, reject) => {
          getLogin(params)
            .then((response) => {
              const { status, message, data } = response || {};
              if (status === 200) {
                // 存入 参数: 1.调用的值 2.所要存入的数据
                commit('SET_USER_INFO', data);
                commit('SET_TOKEN', data.uuid);
                commit('SET_USERID', data.id);
                commit('SET_ROLES', data.roles);
                commit('SET_NAME', data.realName);
                commit('SET_TENANTID', data.tenantId || 1);
                setToken(data.uuid);
                db.save('userInfo', data);
                db.save('tenantId', data.tenantId || 1);
                localStorage.setItem('loginToken', data.uuid);
                resolve(data);
              } else {
                // ElementUI.Message.error(message);  // axios拦截统一提示了
              }
            })
            .catch((error) => {
              // ElementUI.Message.error(error.message); // axios拦截统一提示了
              reject(error);
            });
        });
      },}
Die Aufrufmethode dieser Aktion in der Komponente lautet:
 this.$store.dispatch('user/login', postUser)
            .then(res => {
               // .............
              })// 我这里的login方法写在了user.js这个module里 所以这里调用是 user/login// 下面会讲到module

Getter verarbeiten den Status

Getter entsprechen berechneten Eigenschaften, werden zum Verarbeiten von Statusdaten verwendet und verfügen über zwei Standardparameter: den ersten Standard Der Parameter ist „state“ und der zweite Standardparameter ist „getters“.

const getters={
  plusCount(state){
    return state.count + 1
  },
 //获取state中状态数据对象,和获取getters模块中plusCount数据
  totalCount(state,getters){
    return getters.plusCount + state.count  }}

Der Codeausschnitt zum Aufrufen dieser Methode in der Komponente lautet:

this.$store.getters.totalCount()

Holen Sie sich den Vuex-Status

in der Vue-Komponente. Wenn er sich von

ändert, werden die berechneten Eigenschaften für reaktionsfähige Aktualisierungen erneut abgerufen.

 computed: {
        count: function(){
            return this.$store.state.count        }
    },

那么对于以上的store我们就简单介绍完了,相信大家看完后对于vuex会有一定的理解。那么这个时候我们要想,是不是使用this.$store.statethis.$store.getters.xxx感到麻烦呢?下面我们介绍另一种引入state和getters的方式

辅助函数 mapState 和 mapGetters

对于上述的在组件中引用state和getters的方法是不是感到麻烦呢?使用mapState你将会感受到便利。
组件中这样使用

//首先我们需要先将辅助函数引入import { mapGetters,mapState } from 'vuex'
 export default {
 computed: {

  // 使用对象展开运算符将 getter 混入 computed 对象中
  ...mapGetters( ['plusCount','totalCount'] )

 // 使用对象展开运算符将 state 混入 computed 对象中
    ...mapState( ['userInfo','count'] )

 },methods:{
    getData(){
        // 这里就能直接使用了  直接使用state 和getters里的数据
        //  this.userInfo 
        // this.plusCount
        
    }}}

Module子模块化管理

store文件夹下的index.js代码如下

import Vue from 'vue'import Vuex from 'vuex'import getters from './getters'Vue.use(Vuex)const modulesFiles = require.context('./modules', true, /\.js$/)const modules = modulesFiles.keys().reduce((modules, modulePath) => {
  const moduleName = modulePath.replace(/^\.\/(.*)\.\w+$/, '$1')
  const value = modulesFiles(modulePath)
  modules[moduleName] = value.default
  return modules}, {})const store = new Vuex.Store({
  modules,
  getters})export default store

文件目录如图

Leicht zu verstehen! Ausführliche Erläuterung der staatlichen Lagerverwaltung von VUEX

举例 api.js

import { getKey, getLogin, logout, getInfo } from '@/api/user';import { setToken, removeToken } from '@/utils/auth';import db from '@/utils/localstorage';import router, { resetRouter } from '@/router';import ElementUI from 'element-ui';const state = {
  userId: '',
  token: '',
  name: '',
  avatar: '',
  introduction: '',
  roles: [],
  tenantId: 1,
  userInfo: null
  // roles: ['9999']};const mutations = {
  SET_TOKEN: (state, token) => {
    state.token = token;
  },
  SET_USERID: (state, userId) => {
    state.userId = userId;
  },
  SET_NAME: (state, name) => {
    state.name = name;
  },
  SET_ROLES: (state, roles) => {
    state.roles = roles;
  },
  SET_TENANTID: (state, roles) => {
    state.tenantId = roles;
  },
  SET_USER_INFO: (state, userInfo) => {
    state.userInfo = userInfo;
  }};const actions = {
  // 获取密钥
  getKey({ commit }) {
    return new Promise((resolve, reject) => {
      getKey()
        .then((response) => {
          resolve(response);
        })
        .catch((error) => {
          reject(error);
        });
    });
  },

  // 用户登录
  login({ commit }, userInfo) {
    // const { username, password } = userInfo;
    const params = userInfo;
    params.userName = userInfo.userName.trim()
    return new Promise((resolve, reject) => {
      // console.log(username, password);
      // setToken(state.token)
      // localStorage.setItem('loginToken', state.token)
      getLogin(params)
      // getLogin({ userName: username.trim(), password: password })
        .then((response) => {
          const { status, message, data } = response || {};
          if (status === 200) {
            // 存入 参数: 1.调用的值 2.所要存入的数据
            commit('SET_USER_INFO', data);
            commit('SET_TOKEN', data.uuid);
            commit('SET_USERID', data.id);
            commit('SET_ROLES', data.roles);
            commit('SET_NAME', data.realName);
            commit('SET_TENANTID', data.tenantId || 1);
            setToken(data.uuid);
            db.save('userInfo', data);
            db.save('tenantId', data.tenantId || 1);
            localStorage.setItem('loginToken', data.uuid);
            resolve(data);
          } else {
            // ElementUI.Message.error(message);  // axios拦截统一提示了
          }
        })
        .catch((error) => {
          // ElementUI.Message.error(error.message); // axios拦截统一提示了
          reject(error);
        });
    });
  },

  // 获取用户信息
  getInfo({ commit, state }) {
    return new Promise((resolve, reject) => {
      getInfo(state.token)
        .then((response) => {
          const { data } = response;
          data.roles = response.data.rights.map(String);
          if (!data) {
            reject('验证失败,请重新登录。');
          }
          const loginMessage = {
            memberId: data.id,
            userName: data.name,
            userTel: data.mobile,
            realName: data.realName,
            incorCom: data.incorCom,
            virtualCor: data.virtualCor,
            deptId: data.deptId,
            deptpath: data.deptpath,
            deptName: data.deptName          };
          localStorage.setItem('loginMessage', JSON.stringify(loginMessage));
          const { id, roles, realName } = data;
          // 角色必须是非空数组!
          if (!roles || roles.length <= 0) {
            reject('getInfo: 角色必须是非空数组!');
          }
          commit('SET_USERID', id);
          commit('SET_ROLES', roles);
          commit('SET_NAME', realName);
          localStorage.setItem('userRights', roles);
          // commit('SET_AVATAR', avatar)
          // commit('SET_INTRODUCTION', introduction)
          resolve(data);
        })
        .catch((error) => {
          reject(error);
        });
    });
  },

  // 用户登出
  logout({ commit, state }) {
    return new Promise((resolve, reject) => {
      logout(state.token)
        .then(() => {
          commit('SET_TOKEN', '');
          commit('SET_ROLES', []);
          db.remove('router');
          removeToken();
          resetRouter();
          resolve();
        })
        .catch((error) => {
          reject(error);
        });
    });
  },

  // 删除token
  resetToken({ commit }) {
    return new Promise((resolve) => {
      commit('SET_TOKEN', '');
      commit('SET_ROLES', []);
      removeToken();
      resolve();
    });
  },

  // 动态修改权限
  changeRoles({ commit, dispatch }, role) {
    return new Promise(async(resolve) => {
      const token = role + '-token';

      commit('SET_TOKEN', token);
      setToken(token);

      const { roles } = await dispatch('getInfo');
      console.log(roles, 'rolesrolesroles');
      resetRouter();

      // 根据角色生成可访问路由映射
      const accessRoutes = await dispatch('permission/generateRoutes', roles, {
        root: true
      });

      // 动态添加可访问路由
      router.addRoutes(accessRoutes);

      // 重置已访问视图和缓存视图
      dispatch('tagsView/delAllViews', null, { root: true });

      resolve();
    });
  }};export default {
  namespaced: true,
  state,
  mutations,
  actions};

这样后可以按功能分module使用

页面中调用就是

// 使用mutationsthis.$store.commit(&#39;api/SET_T&#39;, keys);// 使用actionsthis.$store.dispatch(&#39;user/login&#39;, postUser).then(res => {})// 如果没有分module // 那就是 this.$store.commit(&#39;SET_T&#39;, keys);// 直接调用方法

写完自己也感觉好简单噢(⊙-⊙)

不明白的童鞋在评论区留言咯 ヾ(•ω•`)o

Das obige ist der detaillierte Inhalt vonLeicht zu verstehen! Ausführliche Erläuterung der staatlichen Lagerverwaltung von VUEX. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:jianshu.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen