文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

面试官: 如何让localStorage支持过期时间设置?

2024-12-02 13:12

关注

聊到 localStorage 想必熟悉前端的朋友都不会陌生, 我们可以使用它提供的 getItem, setItem, removeItem, clear 这几个 API 轻松的对存储在浏览器本地的数据进行读,写, 删操作, 但是相比于 cookie, localStorage 唯一美中不足的就是不能设置每一个键的过期时间。

localStorage 属性允许我们访问一个 Document 源(origin)的对象 Storage;存储的数据将保存在浏览器会话中。localStorage 类似 sessionStorage,但其区别在于:存储在 localStorage 的数据可以长期保留;而当页面会话结束——也就是说,当页面被关闭时,存储在 sessionStorage 的数据会被清除 。

我们还应注意,localStorage 中的键值对总是以字符串的形式存储。

问题描述

在实际的应用场景中, 我们往往需要让 localStorage 设置的某个 key 能在指定时间内自动失效, 所以基于这种场景, 我们如何去解决呢?

1. 初级解法

对于刚熟悉前端的朋友, 可能会立马给出答案:

  1. localStorage.setItem('dooring''1.0.0'
  2. // 设置一小时的有效期 
  3. const expire = 1000 * 60 * 60; 
  4. setTimeout(() => { 
  5.   localStorage.setItem('dooring'''
  6. }, expire) 

当然这种方案能解决一时的问题, 但是如果要设置任意键的有效期, 使用这种方案就需要编写多个定时器, 维护成本极高, 且不利于工程化复用。

2. 中级解法

前端工程师在有一定的工作经验之后, 往往会去考虑工程化和复用性的问题, 并对数据结构有了一定的了解, 所以可能会有接下来的解法:

  1. 用localStorage存一份{key(键): expire(过期时间)}的映射表
  2. 重写localStorage API, 对方法进行二次封装

类似的代码如下:

  1. const store = { 
  2.   // 存储过期时间映射 
  3.   setExpireMap: (key, expire) => { 
  4.     const expireMap = localStorage.getItem('EXPIRE_MAP') || "{}" 
  5.     localStorage.setItem( 
  6.       'EXPIRE_MAP',  
  7.       JSON.stringify({ 
  8.       ...JSON.parse(expireMap), 
  9.       key: expire 
  10.     })) 
  11.   }, 
  12.   setItem: (key, value, expire) => { 
  13.     store.setExpireMap(key, expire) 
  14.     localStorage.setItem(key, value) 
  15.   }, 
  16.   getItem: (key) => { 
  17.     // 在取值之前先判断是否过期 
  18.     const expireMap = JSON.parse( 
  19.       localStorage.getItem('EXPIRE_MAP') || "{}" 
  20.     ) 
  21.     if(expireMap[key] && expireMap[key] < Date.now()) { 
  22.       return localStorage.getItem(key
  23.     }else { 
  24.       localStorage.removeItem(key
  25.       return null 
  26.     } 
  27.   } 
  28.   // ... 

眨眼一看这个方案确实解决了复用性的问题, 并且不同团队都可以使用这个方案, 但仍然有一些缺点:

3. 高级解法

为了减少维护成本和空间占用, 并支持一定的灵活控制和容错能力, 我们又应该怎么做呢?

这里笔者想到了两种类似的方案:

  1. 将过期时间存到 key 中, 如 dooring|6000, 每次取值时通过分隔符“|”来将 key 和 expire 取出, 进行判断
  2. 将过期时间存到 value 中, 如 1.0.0|6000, 剩下的同1

为了更具有封装性和可靠性, 我们还可以配置不同状态下的回调, 简单实现如下:

  1. const store = { 
  2.   preId: 'xi-'
  3.   timeSign: '|-door-|'
  4.   status: { 
  5.     SUCCESS: 0, 
  6.     FAILURE: 1, 
  7.     OVERFLOW: 2, 
  8.     TIMEOUT: 3, 
  9.   }, 
  10.   storage: localStorage || window.localStorage, 
  11.   getKey: function (key: string) { 
  12.     return this.preId + key
  13.   }, 
  14.   setfunction ( 
  15.     key: string, 
  16.     value: string | number, 
  17.     time?: Date & number, 
  18.     cb?: (status: number, key: string, value: string | number) => void, 
  19.   ) { 
  20.     let _status = this.status.SUCCESS, 
  21.       _key = this.getKey(key), 
  22.       _time; 
  23.     // 设置失效时间,未设置时间默认为一个月 
  24.     try { 
  25.       _time = time 
  26.         ? new Date(time).getTime() || time.getTime() 
  27.         : new Date().getTime() + 1000 * 60 * 60 * 24 * 31; 
  28.     } catch (e) { 
  29.       _time = new Date().getTime() + 1000 * 60 * 60 * 24 * 31; 
  30.     } 
  31.     try { 
  32.       this.storage.setItem(_key, _time + this.timeSign + value); 
  33.     } catch (e) { 
  34.       _status = this.status.OVERFLOW; 
  35.     } 
  36.     cb && cb.call(this, _status, _key, value); 
  37.   }, 
  38.   get: function ( 
  39.     key: string, 
  40.     cb?: (status: number, value: string | number | null) => void, 
  41.   ) { 
  42.     let status = this.status.SUCCESS, 
  43.       _key = this.getKey(key), 
  44.       value = null
  45.       timeSignLen = this.timeSign.length, 
  46.       that = this, 
  47.       index
  48.       time
  49.       result; 
  50.     try { 
  51.       value = that.storage.getItem(_key); 
  52.     } catch (e) { 
  53.       result = { 
  54.         status: that.status.FAILURE, 
  55.         value: null
  56.       }; 
  57.       cb && cb.call(this, result.status, result.value); 
  58.       return result; 
  59.     } 
  60.     if (value) { 
  61.       index = value.indexOf(that.timeSign); 
  62.       time = +value.slice(0, index); 
  63.       if (time > new Date().getTime() || time == 0) { 
  64.         value = value.slice(index + timeSignLen); 
  65.       } else { 
  66.         (value = null), (status = that.status.TIMEOUT); 
  67.         that.remove(_key); 
  68.       } 
  69.     } else { 
  70.       status = that.status.FAILURE; 
  71.     } 
  72.     result = { 
  73.       status: status, 
  74.       value: value, 
  75.     }; 
  76.     cb && cb.call(this, result.status, result.value); 
  77.     return result; 
  78.   }, 
  79.   // ... 
  80. }; 
  81.  
  82. export default store; 

这样, 我们就实现了每个 key 都有独立的过期时间, 并且对不同的操作结果可以轻松的进行状态管控啦~

4. 骨灰级解法

当然, 骨灰级解法是直接使用 xijs 这个 javascript 工具库, 因为我已经将上述完整实现方案封装到该库中了, 我们只需要使用如下的方案, 就能轻松使用具有过期时间的强大的 localStorage 方法啦 :

  1. //  先安装 yarn add xijs 
  2. import { store } from 'xijs'
  3. // 设置带有过期时间的key 
  4. store.set('name''dooring'Date.now() + 1000); 
  5. console.log(store.get('name')); 
  6. setTimeout(() => { 
  7.   console.log(store.get('name')); 
  8. }, 1000); 
  9.  
  10. // 设置成功后的回调 
  11. store.set('dooring''xuxiaoxi'Date.now() + 1000, (status, key, value) => { 
  12.   console.log('success'); 
  13. }); 

同时 xijs 还在持续扩充更有用的工具函数, 让业务开发更高效. 目前已集成了如下工具函数:

本文转载自微信公众号「趣谈前端」

【编辑推荐】

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区
  2. 万字长文 | Linux C/C++ 后台服务器开发学习路线
  3. Log4j史诗级漏洞,我们这些小公司能做些什么?
  4. Spring Boot 应对 Log4j2 注入漏洞官方指南
  5. 八个好用到爆的Python实用技巧,不用吃亏半年
  6. iOS15.2准正式版,今日正式发布!加入四大更新

 

来源:趣谈前端内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯