vuex状态持久化 方案一

vuex状态持久化 方案一

前言

我们都知道Vuex是一个状态管理器,而他的缺点也很明确,在页面刷新之后,Vuex中的状态都会被重置,这对于一些不想被重置的状态数据而言,
是一个不好的表现。如果是完全用Vue构建的 app 项目的话,则不需要考虑这些,因为在 app 中,不存在刷新浏览器的操作。

当然,如果是混合开发的,那还是有一些可能的,比如 app 端重新加载 webview 的话,那也是等同于刷新浏览器的操作了,这个时候 Vuex 也会被重置。

而今天要讨论的就是让Vuex的状态持久化,当然,这只是其中一个方案,这里我们需要配合本地存储来达到我们的目标。

问题

  • 并不是所有状态都需要存入本地缓存
  • 重置默认值,并不是所有的状态默认值都是''

实现

Vuex Demo

首先我们初始化一个vue项目:

vue init webpack vuexDemo

初始化成功后,我们通过yarn安装vuex

yarn add vuex

安装成功后,我们在项目根目录src建立一个store文件夹,该文件夹用于存放vuex的内容,结构入下:

store
├── state.js            # vuex状态集合
├── getter.js           # state的派生状态 可对state做些过滤或者其他操作
├── action.js           # 异步mutation操作
├── mutation.js         # 修改state状态
├── mutation-type.js    # mutation的类型
├── index.js            # vuex主文件

我们对各个文件加入点简单的内容:

// state.js
const state = {
  count: 0
}

export default state

// mutation-type.js
export const SET_COUNT = 'SET_COUNT'

// mutation.js
import * as type from './mutation-type'

const mutation = {
  [type.SET_COUNT](state, data) {
    state.count = data
  }
}

export default mutation

// index.js
import Vue from 'vue'
import Vuex from 'vuex'
// import * as actions from './action'
// import * as getters from './getter'
import state from './state'
import mutations from './mutation'

Vue.use(Vuex)

export default new Vuex.Store({
  actions,
  getters,
  state,
  mutations
})

这里之所以没有引入getteraction,一个是因为我们取state中的count是直接取,并没有对其做什么特别的操作,所以getter中就省略了。

action它提交的是一个mutation,而且它和mutation的区别在于:

  • mutation是同步的,action可以包含异步操作
  • mutation直接修改state,而action提交的是mutation,然后再让mutation去修改state
  • action可以一次提交多个mutation

我们现在就是个简单的修改state,所以就先不管action

然后在vue模板中:

<template>
  <div class="hello">
    <h1>{{ msg }}</h1>
    <h3>{{ count }}</h3>
    <button @click="addStore">添加</button>
  </div>
</template>

<script>
import { mapState, mapMutations } from 'vuex'
export default {
  name: 'HelloWorld',
  computed: {
    ...mapState(['count'])
  },
  data() {
    return {
      msg: 'Welcome to Your Vue.js App'
    }
  },
  methods: {
    addStore() {
      this.SET_COUNT(2)
    },
    ...mapMutations(['SET_COUNT'])
  }
}
</script>

这样就完成了一个简单的vuex例子了,当点击添加按钮的时候,界面上的0就会变成2,并且如果装有vue-devtools的话,
也能在vuex那一栏看到count的数值也变成了2,这里就不放动图演示了。

持久化

接下来就是我们的关键内容了,想要让vuex持久化,自然离不开本地存储localStorage,我们往state里加些内容:

// state.js
const str = window.localStorage

const state = {
  count: 0,
  account: str.getItem('account') ? str.getItem('account') : ''
}

我们加入了一个account属性,这里表示如果缓存中有account的话就从缓存中取,没有则为空。

然后我们也同样设置下mutationmutation-type

// mutation-type.js
export const SET_COUNT = 'SET_COUNT'

export const SET_ACCOUNT_0 = 'SET_ACCOUNT_0'

// mutation.js
import * as type from './mutation-type'

const mutation = {
  [type.SET_COUNT](state, data) {
    state.count = data
  },
  [type.SET_ACCOUNT_0](state, account) {
    state.account = account
  }
}

export default mutation

我们在定义mutation-type的时候,在尾部多加了个_0用来表示,该字段是存入缓存中的。

但是我们不会选择在mutation中去做缓存操作,毕竟我个人认为不适合在mutation中做过多的逻辑操作,我们选择将这部分逻辑操作放在action中:

// action.js
import * as type from './mutation-type'
const str = window.localStorage

/**
 * 缓存操作
 */
export const withCache = ({ commit }, { mutationType, data }) => {
  commit(mutationType, data)
  // 是不是以_0结尾 是的话表示需要缓存
  if (~mutationType.indexOf('_0')) {
    setToStorage(mutationType, data)
  }
}

// 正则太烂。。。就先这么写着了
const reg = /(SET_)(\w+)(_0)/
function setToStorage(type, data) {
  let key = type.match(reg)[2].toLowerCase()
  if (typeof data === 'string') str.setItem(key, data)
  else {
    let formatData = JSON.stringify(data)
    str.setItem(key, formatData)
  }
}

上面这段代码解决几个问题:

  • 因为在action中我不知道存储的目标属于哪个type,所以将其当做参数传入
  • 存入缓存的key我们取的是SET_xxx中的xxx(小写),尽量保持和state的字段名称一致。比如SET_A->aSET_B_0->b
  • 本地缓存存Object类型会变成[Object object],所以针对Object类型的数据,我们需要将其转成字符串再存入

提示
记得将actionindex.js引入

然后我们在vue模板中,先引入mapActions,然后进行使用:

<script>
import { mapState, mapMutations, mapActions } from 'vuex'
export default {
  name: 'HelloWorld',
  computed: {
    ...mapState(['count'])
  },
  data() {
    return {
      msg: 'Welcome to Your Vue.js App'
    }
  },
  methods: {
    addStore() {
      let account = { user: 'Randy', age: 22 }
      this.withCache({ mutationType: 'SET_COUNT', data: 2 })
      this.withCache({ mutationType: 'SET_ACCOUNT_0', data: account })
    },
    ...mapMutations(['SET_COUNT']),
    ...mapActions(['withCache'])
  }
}
</script>

直接调用withCache,传入mutationTypedatawithCache会根据mutationType判断哪些是需要存入缓存的,哪些是不需要的。

当然,不需要存入缓存的,也可以直接调用mapMutations中的方法直接操作。

现在有个问题,就是我缓存存入Object类型的是字符串类型,所以我state中的对应数据也是字符串类型的,在模板中不利于使用,怎么办?这时候就可以使用getter了,我们在getter中将其转成Object类型即可:

// getter.js
export const getAccount = state => {
  let account = state.account
  if (typeof account === 'string' && !!account) return JSON.parse(account)
  else return account
}

提示
上面的判断还不够完整,应该还要判断是否是 JSON 字符串类型,是的话再进行JSON.parse操作,不然普通的字符串类型会报错,这个自行改善。

提示
记得将getterindex.js引入

然后在模板中使用mapGetters引入:

<template>
  <div>
    ...
    {{getAccount.user}}
    ...
  </div>
</template>
<script>
import { mapState, mapMutations, mapActions, mapGetters } from 'vuex'
export default {
  name: 'HelloWorld',
  computed: {
    ...mapState(['count']),
    ...mapGetters(['getAccount'])
  },
  data() {
    return {
      msg: 'Welcome to Your Vue.js App'
    }
  },
  methods: {
    addStore() {
      let account = { user: 'Randy', age: 22 }
      this.withCache({ mutationType: 'SET_COUNT', data: 2 })
      this.withCache({ mutationType: 'SET_ACCOUNT_0', data: account })
    },
    ...mapMutations(['SET_COUNT']),
    ...mapActions(['withCache'])
  }
}
</script>

到这基本就结束了,现在刷新浏览器,那些需要持久化的属性就不会被重置了。

可是真的就结束了吗?那么如果我要将缓存的数据给清空或者重置呢?因为state中每个属性的默认值都是不一样的,可能为''0false等各种类型的,那该怎么办?某问题啦~

状态重置

可以复制出一份state作为它的初始默认值,比如在store新建一个default_state.js

// default_state.js
const default_state = {
  count: 0,
  account: ''
}

export default default_state

然后我们定义一个类型为RESET_ALL_STATEmutation

// mutation-type.js
export const RESET_ALL_STATE = 'RESET_ALL_STATE'

// mutation.js
import * as type from './mutation-type'

const mutation = {
  [type.SET_COUNT](state, data) {
    state.count = data
  },
  [type.SET_ACCOUNT_0](state, account) {
    state.account = account
  },
  [type.RESET_ALL_STATE](state, data) {
    state[`${data.state}`] = data.value
  }
}

export default mutation

最后我们在action中定义一个重置的操作resetAllState

// action.js
import * as type from './mutation-type'
import default_state from './default_state'
const str = window.localStorage

/**
 * 重置所有状态
 */
export const resetAllState = ({ commit }) => {
  // 循环默认state 设置初始值
  Object.keys(default_state).forEach(state => {
    commit(type.RESET_ALL_STATE, { state, value: default_state[state] })
  })
  // 将有缓存的数据清空
  Object.keys(type).forEach(typeItem => {
    if (~typeItem.indexOf('_0')) clearStorage(type[typeItem])
  })
}

const reg = /(SET_)(\w+)(_0)/
function clearStorage(type) {
  let key = type.match(reg)[2].toLowerCase()
  str.removeItem(key)
}

完整action.js

import * as type from './mutation-type'
import default_state from './default_state'
const str = window.localStorage

/**
 * 缓存操作
 */
export const withCache = ({ commit }, { mutationType, data }) => {
  commit(mutationType, data)
  if (~mutationType.indexOf('_0')) {
    // 需要缓存
    setToStorage(mutationType, data)
  }
}

/**
 * 重置所有状态
 */
export const resetAllState = ({ commit }) => {
  // 循环默认state 设置初始值
  Object.keys(default_state).forEach(state => {
    commit(type.RESET_ALL_STATE, { state, value: default_state[state] })
  })
  // 将有缓存的数据清空
  Object.keys(type).forEach(typeItem => {
    if (~typeItem.indexOf('_0')) clearStorage(type[typeItem])
  })
}

const reg = /(SET_)(\w+)(_0)/
function setToStorage(type, data) {
  let key = type.match(reg)[2].toLowerCase()
  if (typeof data === 'string') str.setItem(key, data)
  else {
    let formatData = JSON.stringify(data)
    str.setItem(key, formatData)
  }
}

function clearStorage(type) {
  let key = type.match(reg)[2].toLowerCase()
  str.removeItem(key)
}

vue模板中的完整使用:

<template>
  <div class="hello">
    <h1>{{ msg }}</h1>
    <h3>{{ count }}</h3>
    <h4>{{ getAccount.user }}</h4>
    <button @click="addStore">添加</button>
    <button @click="clearStore">清空</button>
  </div>
</template>

<script>
import { mapGetters, mapState, mapMutations, mapActions } from 'vuex'
export default {
  name: 'HelloWorld',
  computed: {
    ...mapState(['count']),
    ...mapGetters(['getAccount'])
  },
  data () {
    return {
      msg: 'Welcome to Your Vue.js App'
    }
  },
  methods: {
    addStore () {
      let account = { user: 'Randy', age: 22 }
      this.withCache({ mutationType: 'SET_COUNT', data: this.msg })
      this.withCache({ mutationType: 'SET_ACCOUNT_0', data: account })
    },
    clearStore () {
      this.resetAllState()
    },
    ...mapMutations(['SET_COUNT']),
    ...mapActions(['withCache', 'resetAllState'])
  }
}
</script>

好了,现在是真的结束了。

总结
可能代码有些乱,但是大致的思路我想应该还是都能理解的。
如果还有有其他vuex的持久化方式,还会继续更新的。
也欢迎大家一同思考。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • Vuex 概念篇 Vuex 是什么? Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式。它采用集中式...
    Junting阅读 3,066评论 0 43
  • 安装 npm npm install vuex --save 在一个模块化的打包系统中,您必须显式地通过Vue.u...
    萧玄辞阅读 2,931评论 0 7
  • Vuex Vuex是一个专门为Vue.js应用所设计的集中式状态管理架构,它借鉴了Flux和Redux的设计思想,...
    JunChow520阅读 1,567评论 2 0
  • vuex是一个状态管理模式,通过用户的actions触发事件,然后通过mutations去更改数据(你也可以说状态...
    Ming_Hu阅读 2,021评论 3 3
  • 一起去幼儿园的第七天。 每天早上孩子们的情绪会决定他们一整天的学习效率,玩的状态。昨天早上因为我的状态不好,大姨妈...
    瑞玛和麦琪阅读 156评论 0 0