芭比 Q 了,vuex-persistedstate 官宣停止维护……

图片

来源丨程序员Sunday(ID:gh_255b41b1f634)

vuex-persistedstate 是一个 基于 vuex 的 状态缓存工具 ,它可以让我们 刷新页面时持久化 state 中的状态。

不过,这个库在 3年前 已经 停止维护了,当它配合 Vue3 + Vuex4 进行使用时会出现一些奇怪的错误。

因此,我就简单实现了一个 vuex-plugin-persistedstate 用于解决 Vue3 + Vuex4 的状态持久化问题。实现了之后,就一直没有管它。

让我没想到的是,昨天登录 npm 发现竟然还有 200 多的下载量:

这证明大家对于 vuex-persistedstate 的需求存在的。

因此,今天就借助这个库的代码,讲解下 vuex-persistedstate 的原理,让大家也可以花 3 分钟的时间 实现一个 vuex-persistedstate。

persistedstate 的原理

persistedstate 的原理其实非常简单,核心是两个点:

  • 如何把 state 中的数据进行持久化

  • 如何把持久化的数据赋值给 state

  • 1.1 如何把 state 中的数据进行持久化

    想要把 state 中的数据进行持久化,说白了就是:监听 mutation 事件,在每次 mutation 修改数据后,同步数据到 localStorage 中

    那么如何监听 mutation 事件 呢?

    根据 vuex 文档描述,我们可以直接通过 store.subscribe 方法监听 mutation 提交后的操作:

    1.2 如何把持久化的数据赋值给 state

    从 localStorage 中获取数据非常简单,问题在于 如何保证在刷新页面后,从 localStorage 中获取数据呢?

    因为 vuex 本质上是一个单例的对象,该对象保存在内存中。即:只有页面刷新时,vuex 才会重新执行初始化操作。

    所以,根据这个概念,我们可知:只要执行了初始化操作,那么就以为着内存缓存消失,就需要从从 localStorage 中获取数据了

    因此,我们只需要在 plugin 触发时,读取数据即可

    代码实现

    首先,创建一个基于 ts 的项目,大致结构如下(核心关注 src 中的代码结构):

    根据结构可知,整体的代码非常简单(src 中的代码)一共只有 4 个文件。

    2.1 入口文件 index.ts

    在 index.ts 中,我们需要按照 vuex-plugin 的要求,构建一个基本的函数。

    根据原理中描述,我们需要再这里做两件事情:

  • 只要该函数执行,就从缓存中读取数据

  • 利用 store.subscribe 监听 mutation 行为

  • import { Options, defaultOptions } from'./core/options'import { MutationPayload, Store } from'vuex'import { matchPaths } from'./core/persistedstate'let catchData: object = {}exportdefaultfunction VuexPersistedstate<State>({  key = defaultOptions.key,  paths = defaultOptions.paths,  storage = defaultOptions.storage,  fetchBeforeUse = defaultOptions.fetchBeforeUse,  fetchBeforeUseFn = defaultOptions.fetchBeforeUseFn}: Options<State> = defaultOptions) {// 读取缓存文件if (fetchBeforeUse) {    catchData = fetchBeforeUseFn(key, storage)  }return(store: Store<State>) => {    // 存储缓存数据    for (const key in catchData) {      if (Object.prototype.hasOwnProperty.call(catchData, key)) {        const value = catchData[key]        store.commit(key, value)      }    }    // 每次 mutation 后接收通知    // { type, payload }    store.subscribe((mutation: MutationPayload, state: State) => {      if (matchPaths(paths, mutation)) {        catchData[mutation.type] = mutation.payload        storage.setItem(key, catchData)      }    })  }}

    在这里,我们会用到 core 中的一些依赖库。

    2.2:options 可配置参数

    core/options 中的代码主要提供了 可配置参数,所以核心由 接口 + 默认配置对象 组成

    import storage, { Storage } from'./storage'exportinterface Options<State> {/**   * localStorage 保存的 key   */  key: string/**   * 缓存模块名称   * 不通过意味着缓存所有   * 仅传递指定的缓存   */  paths: string[]/**   * storage   */  storage: Storage/**   * 是否预取数据   */  fetchBeforeUse: boolean/**   * 预取数据的默认方法   */  fetchBeforeUseFn: (key: string, storage: Storage) =>any}exportconst defaultOptions: Options<object> = {  key: 'VUEX-PERSISTEDSTATE',  paths: [],  fetchBeforeUse: true,  fetchBeforeUseFn(key: string, storage: Storage) {    return storage.getItem(key)  },  storage}

    2.3:storage 持久化逻辑

    core/storage 中的代码主要提供了 持久化逻辑,所以核心由 几个沟通 localStorage 的方法 组成

    export interface Storage {  getItem: (key: string) => object  setItem: (key: string, value: any) =>void  removeItem: (key: string) =>void}exportconst getItem = (key: string): object => {const val = JSON.parse(window.localStorage.getItem(key) asstring)if (!val) {    return {}  }return val.value || {}}exportconst setItem = (key: string, value: any) => {let val: object = {    value  }let valStr = JSON.stringify(val)window.localStorage.setItem(key, valStr)}exportconst removeItem = (key: string) => {window.localStorage.removeItem(key)}const storage: Storage = {  getItem,  setItem,  removeItem}exportdefault storage

    2.4:persistedstate 匹配逻辑

    因为 vuex 中提供了 module 的概念,所以在触发 mutations 时可能会存在 路径 的概念。

    因此,需要在 core/persistedstate 中的进行路径解析

    import { MutationPayload } from'vuex'/** * 确定当前匹配是否基于路径的状态 */exportfunction matchPaths(  paths: string[],  mutation: MutationPayload): boolean {if (paths.length === 0) {    returntrue  }const moduleName = mutation.type.split('/')[0]if (!moduleName) {    returnfalse  }return paths.includes(moduleName)}

    总结

    那么到这里,一个极简的 vuex-plugin-persistedstate 就实现完成了。足以满足,大多数情况下的 vuex 持久化存储逻辑。是不是非常简单呢

    图片

      推荐阅读:

  • 未来 Vue3 真的可以无处不在吗?

  • 分享一些 Vue 实用且常用的开发工具库

  • 2024 年 GitHub 上 Star 数增长最快的 Vue 项目

  • Vue3 的 Teleport 是个性能利器,为啥大家都死活不用?

  • 【vue3】手撸 defineModel 实现防抖、多字段转换等功能

  • (0)
    wd123_cnwd123_cn
    上一篇 2025年3月31日 上午10:38
    下一篇 2025年3月31日 上午10:40

    相关文章

    • 59岁追梦不晚!桑德拉·瓦尔斯圆梦百老汇,诠释多元角色

      在追逐梦想的道路上,年龄从来不是障碍。59岁的桑德拉·瓦尔斯用自己的经历完美诠释了这一点。在演艺圈摸爬滚打数十年后,她终于实现了登上百老汇舞台的梦想,在音乐剧《Real Women Have Curves》(真实的女性拥有曲线)中首次亮相。 多年耕耘,终获认可 桑德拉·瓦尔斯并非一夜成名。在踏上百老汇舞台之前,她已经在演艺界默默耕耘了几十年。她曾在乐队中表演…

      2025年4月7日
    • 研究发现:健康植物性饮食或能降低炎症性肠病风险,减少手术需求

      炎症性肠病 (IBD) 并非单一疾病,而是一个涵盖多种肠道疾病的总称,其中包括克罗恩病和溃疡性结肠炎。 该疾病的发生机制是免疫系统错误地攻击肠道细胞,导致炎症,进而引发疼痛、排便习惯改变以及体重下降等症状。 IBD 患者通常可以通过调整饮食来缓解症状,例如避免某些食物或增加某些食物的摄入。 近期一项研究表明,以营养丰富的植物性食物为主的饮食方式,可能有助于降…

      2025年3月25日
    • 阿里云开启近年来规模最大的AI人才校园招聘

      来源 | 财联社 《科创板日报》25日讯,《科创板日报》记者独家获悉,阿里云近日在全球顶尖高校招募AI技术储备人才,为近年来规模最大的AI人才校园招聘。据了解,此次校招面向清华大学、北京大学、浙江大学、麻省理工大学、斯坦福大学等全球顶尖高校,招募大语言模型、多模态理解与生成、模型应用、AI Infra等领域技术人才。同时,项目设置A Star项目和…

      新闻资讯 2025年3月26日
    • 母亲节美味开胃菜指南:让妈妈惊喜的节日味蕾盛宴

      母亲节将至,为辛勤的妈妈们准备一份特别的节日盛宴,怎能少了精致美味的开胃菜?无论您计划一场温馨的早午餐、惬意的下午茶,还是丰盛的晚餐,一份精心准备的开胃菜都能让妈妈感受到您满满的爱意与用心。 早午餐的活力开启 如果您的母亲节计划从早午餐开始,不妨选择一些能唤醒味蕾、开启美好一天的开胃小点。肉桂卷猴子面包,甜蜜诱人,是开启甜蜜一天的完美选择。色彩缤纷的水果串,…

      2025年3月13日
    • 微软将在马来西亚推出首个云区域,投资助其成为东南亚云与AI中心

      路透社吉隆坡消息,微软公司周四宣布,计划在马来西亚推出其首个云区域,其中包括三个数据中心,预计将于今年年中投入运营。此举距离微软宣布在马来西亚投资22亿美元的消息仅过去不到一年。 云区域选址及启动时间 微软马来西亚区总经理Laurence Si在新闻发布会上表示,这些数据中心将构成“马来西亚西部云区域”,选址位于吉隆坡大都会区,预计将于今年第二季度开始运营。…

      2025年3月21日
    • 2025年厨房设计潮流:37个现代奢华厨房灵感

      在厨房设计方面,繁复的装饰已经不再是主流。简洁、时尚的设计理念正受到越来越多的追捧。本文精选了37个现代厨房设计案例,旨在为你在2025年打造梦想中的奢华厨房提供灵感,并附有精美图片供你参考。 来自顶尖设计师的灵感 这些现代厨房设计案例,涵盖了从宽敞空间到紧凑型厨房的各种类型,均出自全国顶尖室内设计师之手。无论你偏爱纯净的白色厨房,还是充满活力的彩色厨房,亦…

      2025年3月21日