自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

芭比 Q 了,Vuex-persistedstate 官宣停止維護(hù)......

開發(fā) 前端
一個極簡的 vuex-plugin-persistedstate 就實現(xiàn)完成了。足以滿足,大多數(shù)情況下的 vuex 持久化存儲邏輯。是不是非常簡單呢??

Hello,大家好,我是 Sunday。

vuex-persistedstate 是一個 基于 vuex 的 狀態(tài)緩存工具 ,它可以讓我們 刷新頁面時持久化 state 中的狀態(tài)。

不過,這個庫在 3年前 已經(jīng) 停止維護(hù)了,當(dāng)它配合 Vue3 + Vuex4 進(jìn)行使用時會出現(xiàn)一些奇怪的錯誤。

因此,我就簡單實現(xiàn)了一個 vuex-plugin-persistedstate 用于解決 Vue3 + Vuex4 的狀態(tài)持久化問題。實現(xiàn)了之后,就一直沒有管它。

讓我沒想到的是,昨天登錄 npm 發(fā)現(xiàn)竟然還有 200 多的下載量:

這證明大家對于 vuex-persistedstate 的需求存在的。

因此,今天就借助這個庫的代碼,講解下 vuex-persistedstate 的原理,讓大家也可以花 3 分鐘的時間 實現(xiàn)一個 vuex-persistedstate。

1.persistedstate 的原理

persistedstate 的原理其實非常簡單,核心是兩個點(diǎn):

  1. 如何把 state 中的數(shù)據(jù)進(jìn)行持久化
  2. 如何把持久化的數(shù)據(jù)賦值給 state

1.1 如何把 state 中的數(shù)據(jù)進(jìn)行持久化

想要把 state 中的數(shù)據(jù)進(jìn)行持久化,說白了就是:監(jiān)聽 mutation 事件,在每次 mutation 修改數(shù)據(jù)后,同步數(shù)據(jù)到 localStorage 中

那么如何監(jiān)聽 mutation 事件 呢?

根據(jù) vuex 文檔描述,我們可以直接通過 store.subscribe 方法監(jiān)聽 mutation 提交后的操作:

圖片圖片

1.2 如何把持久化的數(shù)據(jù)賦值給 state

從 localStorage 中獲取數(shù)據(jù)非常簡單,問題在于 如何保證在刷新頁面后,從 localStorage 中獲取數(shù)據(jù)呢?

因為 vuex 本質(zhì)上是一個單例的對象,該對象保存在內(nèi)存中。即:只有頁面刷新時,vuex 才會重新執(zhí)行初始化操作。

所以,根據(jù)這個概念,我們可知:只要執(zhí)行了初始化操作,那么就以為著內(nèi)存緩存消失,就需要從從 localStorage 中獲取數(shù)據(jù)了

因此,我們只需要在 plugin 觸發(fā)時,讀取數(shù)據(jù)即可

2.代碼實現(xiàn)

首先,創(chuàng)建一個基于 ts 的項目,大致結(jié)構(gòu)如下(核心關(guān)注 src 中的代碼結(jié)構(gòu)):

圖片圖片

根據(jù)結(jié)構(gòu)可知,整體的代碼非常簡單(src 中的代碼)一共只有 4 個文件。

2.1 入口文件 index.ts

在 index.ts 中,我們需要按照 vuex-plugin 的要求,構(gòu)建一個基本的函數(shù)。

根據(jù)原理中描述,我們需要再這里做兩件事情:

  1. 只要該函數(shù)執(zhí)行,就從緩存中讀取數(shù)據(jù)
  2. 利用 store.subscribe 監(jiān)聽 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>) => {
    // 存儲緩存數(shù)據(jù)
    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 可配置參數(shù)

core/options 中的代碼主要提供了 可配置參數(shù),所以核心由 接口 + 默認(rèn)配置對象 組成

import storage, { Storage } from'./storage'

exportinterface Options<State> {
/**
   * localStorage 保存的 key
   */
  key: string
/**
   * 緩存模塊名稱
   * 不通過意味著緩存所有
   * 僅傳遞指定的緩存
   */
  paths: string[]
/**
   * storage
   */
  storage: Storage
/**
   * 是否預(yù)取數(shù)據(jù)
   */
  fetchBeforeUse: boolean
/**
   * 預(yù)取數(shù)據(jù)的默認(rèn)方法
   */
  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 的概念,所以在觸發(fā) mutations 時可能會存在 路徑 的概念。

因此,需要在 core/persistedstate 中的進(jìn)行路徑解析

import { MutationPayload } from'vuex'

/**
 * 確定當(dāng)前匹配是否基于路徑的狀態(tài)
 */
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)
}

總結(jié)

那么到這里,一個極簡的 vuex-plugin-persistedstate 就實現(xiàn)完成了。足以滿足,大多數(shù)情況下的 vuex 持久化存儲邏輯。是不是非常簡單呢?

責(zé)任編輯:武曉燕 來源: 程序員Sunday
相關(guān)推薦

2024-07-29 12:21:12

2024-08-13 13:11:02

2024-05-17 16:04:45

iOS 18蘋果

2015-07-14 09:58:10

2020-12-25 14:55:13

VS Code代碼編程

2023-08-31 22:50:12

微軟開發(fā)

2025-03-07 00:00:10

2018-11-15 11:17:46

智慧城市

2019-03-18 10:01:49

操作系統(tǒng)Android 華為

2025-01-09 09:05:15

2021-05-28 13:44:02

華為MatePad Pro鴻蒙系統(tǒng)

2016-12-09 08:51:18

GoogleDevelopers網(wǎng)站

2024-01-18 15:17:56

谷歌云計算三星

2023-09-08 10:21:21

2022-04-25 10:34:19

云原生直播

2025-04-07 02:21:00

2023-01-31 15:15:14

ChatGPTAI

2019-12-26 09:26:50

青云QingCloud
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號