使用发布订阅模式监听Localstorage值变化

前端项目中经常使用到localstorage本地存储数据,有时候想监听某个值改变然后使用最新值进行业务操作,而不想每次手动处理这些步骤,可以监听值变化的模式进行统一处理。

一、使用发布订阅模式集中管理值

  • events:对象,里面的键代表值名称,值是一个包含很多订阅方法数组。
  • release:每当某个值改变,找到events中对应的值,循环执行数组中的订阅方法。
  • subscribe:订阅的方法,想监听某个值变化,调用方法并传入键名和订阅回调函数,在回调函数中拿到最新值。
  • clear:主要是配置Localstorage.clear()清除所有值。
scss 复制代码
listerer.ts
const publishSubscription = () => {
    const events:{[key:string]:Function[]} = {}

    function release (name:string,param:any){
        if(events[name]){
            events[name].forEach((fn)=>{
                fn(param)
            })
        }
    }

    function subscribe  (name:string,fn:Function)  {
        if(!events[name]){
            events[name] = []
        }
        events[name].push(fn)
    }

    function clear () {
        const keys = Reflect.ownKeys(events)
        if(keys.length){
            for(let key in events){
                if(Object.hasOwnProperty.call(events,key)){
                   events[key].forEach((fn)=>{
                        fn(null)
                    })
                }
            }
        }
    }

    return {
        release,
        subscribe,
        clear
    }
}
export default publishSubscription()

二、改造Localstorage的相关方法

不直接调Localstorage的api,使用包装后的方法,以便进行发布订阅操作。

  • setItem:每次设置值时,就循环执行events中改键对象的数组中的方法。
  • 通过subscribe订阅了该键的地方,就会在回调函数中拿到最新的值。
typescript 复制代码
import PublishSubscription from "./publishSubscription";

const AhjetStorage = {
    setItem: (key: string, value: any) => {
        localStorage.setItem(key, JSON.stringify(value))
        PublishSubscription.release(key,value)
    },
    getItem: (key: string) => {
        return JSON.parse(localStorage.getItem(key) || 'null')
    },
    removeItem: (key: string) => {
        localStorage.removeItem(key)
        PublishSubscription.release(key,null)
    },
    clear: () => {
        localStorage.clear()
        PublishSubscription.clear()
    },
}
export default AhjetStorage

三、效果

typescript 复制代码
const [user,setUser] = useState('默认值')
const setLocal = () => {
    AhjetStorage.setItem("name","ahjet")
}
useEffect(() => {
    return PublishSubscription.subscribe('name',(val:any)=>{
        setUser(val)
    })
}, [])

设置值后,在回调中拿到最新值,再进行对应操作。

相关推荐
奇迹_h3 小时前
打造你的HTML5打地鼠游戏:零基础入门实践
前端
SuperEugene3 小时前
Vue生态精选篇:Element Plus 的“企业后台常用组件”用法扫盲
前端·vue.js·面试
Neptune13 小时前
JavaScript回归基本功之---类型判断--typeof篇
前端·javascript·面试
贾铭3 小时前
如何实现一个网页版的剪映(三)使用fabric.js绘制时间轴
前端·后端
子兮曰4 小时前
后端字段又改了?我撸了一个 BFF 数据适配器,从此再也不怕接口“屎山”!
前端·javascript·架构
万少6 小时前
使用Trae轻松安装openclaw的教程-附带免费token
前端·openai·ai编程
浪浪山_大橙子6 小时前
OpenClaw 十分钟快速,安装与接入完全指南 - 推荐使用trae 官方 skills 安装
前端·人工智能
忆江南6 小时前
iOS 可视化埋点与无痕埋点详解
前端
离开地球表面_997 小时前
金三银四程序员跳槽指南:从简历到面试再到 Offer 的全流程准备
前端·后端·面试
_柳青杨7 小时前
跨域获取 iframe 选中文本?自己写个代理中间层,再也不求后端!
前端