惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

P
Palo Alto Networks Blog
云风的 BLOG
云风的 BLOG
小众软件
小众软件
V
Visual Studio Blog
Cyber Security Advisories - MS-ISAC
Cyber Security Advisories - MS-ISAC
腾讯CDC
Microsoft Security Blog
Microsoft Security Blog
K
Kaspersky official blog
C
Cisco Blogs
The Last Watchdog
The Last Watchdog
宝玉的分享
宝玉的分享
IT之家
IT之家
Cisco Talos Blog
Cisco Talos Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
W
WeLiveSecurity
NISL@THU
NISL@THU
爱范儿
爱范儿
AI
AI
Security Latest
Security Latest
T
The Blog of Author Tim Ferriss
M
MIT News - Artificial intelligence
博客园 - Franky
B
Blog RSS Feed
GbyAI
GbyAI
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
Engineering at Meta
Engineering at Meta
S
Secure Thoughts
Recorded Future
Recorded Future
L
Lohrmann on Cybersecurity
Webroot Blog
Webroot Blog
C
CERT Recently Published Vulnerability Notes
P
Privacy International News Feed
T
Troy Hunt's Blog
L
LangChain Blog
P
Privacy & Cybersecurity Law Blog
Last Week in AI
Last Week in AI
Know Your Adversary
Know Your Adversary
The Cloudflare Blog
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
www.infosecurity-magazine.com
www.infosecurity-magazine.com
P
Proofpoint News Feed
B
Blog
O
OpenAI News
Latest news
Latest news
T
Tor Project blog
Google DeepMind News
Google DeepMind News
F
Fortinet All Blogs
量子位
博客园 - 三生石上(FineUI控件)
Y
Y Combinator Blog

博客园_首页

Linux实操--组管理、权限管理和定时任务 Java + EasyExcel 实现单个接口导出多个Excel Mem0 源码解析系列(二):提示词工程的深度剖析 Openclaw TaskFlow究竟是什么?和普通Skill技能有什么区别 博文阅读密码验证 - 博客园 嘉立创开源:应该是全网MicroPython教程最多的开发板 Hermes Agent 集成实践:从协议到生产 2026年AI编程工具横评:Cursor、Codex、Claude Code、Zed、Windsurf Java程序员必看的RAG入门教程 2026 AI效率神器:Superpowers + Claude Code 保姆级教程 本地大模型部署全攻略:从 0 到 1 玩转 Ollama 【从0到1构建一个ClaudeAgent】内存管理-上下文压缩 .NET 高级开发 | 设计、实现一个事件总线框架 电子小白入门之NE555 3. WorkBuddy:隐藏玩法,一键召唤专家,让 AI 以"专家身份"给你干活 和AI一起搞事情#3:Claude Teammate 游戏开发翻车实录 【OpenClaw】通过 Nanobot 源码学习架构---(7)Memory C# .NET 周刊|2026年3月3期 我在 Debian 11 上把 K8s 单机搭起来了,过程没你想的那么顺(/opt 目录版) 深度学习进阶(七)Data-efficient Image Transformer CLI+Skill搭建浏览器AI自动化框架,告别一切重复枯燥任务 告别Token账单无底洞:OpenClaw本地部署,重塑企业数据主权的唯一解 FastAPI+Vue:文件分片上传+秒传+断点续传,这坑我帮你踩平了! SBTI 爆火后,我做了个程序员版的 CBTI。。已开源 + 附开发过程 多模态检索开始进入工程期:用 Sentence Transformers 搭建可落地的 Multimodal RAG 100多行代码实现一个最简单的Agent(用ReAct) Claude Code 通关手册(八):推荐 5 个 Hooks,代码质量提升 3 倍 老板:“有人截图了!”。安全部门:“收到,马上查暗水印!” - why技术 技术之外,皆是人间 C#/.NET/.NET Core技术前沿周刊 | 第 69 期(2026年4.01-4.12) Snack JSONPath 项目架构分析 Claude Code Buddy 小析:一个非核心功能,如何体现产品的细节完成度 AI新时代下的图床管理方案-Cloudflare图床+MCP+Skills方案指南 化繁为简:顺丰速运App如何通过 HarmonyOS SDK实现专业级空间测量 从零实现富文本编辑器#13-React非编辑节点的内容渲染 AI开发-python-langchain框架(3-23-OpenAI Functions风格Tool Calling智能助手) .NET + AI 进阶实战:基于类的技能开发 - 打造可治理的 Agent 能力模块 【从0到1构建一个ClaudeAgent】规划与协调-技能 上周热点回顾(4.6-4.12) 电子小白的工具三件套:面包板、杜邦线、万能板 单表五亿数据的查询优化 | Mysql、StarRocks 2. WorkBuddy:从“我是谁”到“帮我干活” C# 如何减少代码运行时间:7 个实战技巧 基于HelixToolkit.SharpDX 渲染3D模型 - 笺上知微 从零开始的双臂具身VLA起源及现阶段发展综述 - SkyXZ 记对 xonsh shell 的使用, 脚本编写, 迁移及调优 - pluvium27 受够了Vibe Coding的失控?换个起点,让AI事半功倍 从开始配置漏洞环境到漏洞复现流程 - 難しい 关于10年工作经验的程序员对OpenClaw的实战经验分享以及看法 - 虚无境 Any metadata 的内存布局 C# .NET 周刊|2026年3月2期 - InCerry 我帮你测过了,测试圈排名第二的 Skill 依然很牛逼 Skill Discovery | 无监督技能发现的经典工作总结 - MoonOut PbootCMS 网站内容数量多导致访问慢?这些实用优化方案帮你提速! - 家兴网络技术工作室 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 网站漏洞怎么发现并修复?一篇实用指南(附完整流程) - 家兴网络技术工作室 开了 TUN 模式还是直连?90% 的人都踩过这个坑 Github日报|2026年04月12日 - AI一族 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术 Etsy 把 1000 个 MySQL 分片迁进 Vitess:425TB 数据背后的真正问题不是性能,而是运维规模 MicroPython LVGL基础知识和概念:底层渲染与性能优化 - FreakStudio 数据库草图算法 Python 潮流周刊#146:CPython 引入 Rust 的进展 - 豌豆花下猫 最小生成树 - mofei1116 红日靶场七:从外网入口、容器逃逸到 AD 接管的完整利用链复盘 - YouDiscovered1t 分享四款开源且实用的 Kafka 管理工具 - 追逐时光者 vLLM 权重加载机制全解析:从挑战到理想架构 LCT 学习笔记 - ACehomoxue Avalonia UI 12.0.0 正式发布:架构演进和性能飞跃 - 张善友 当 AI Agent 把调用链拉长,延迟开始成为一门生意 conhost.exe 无法显示 U+2717 - 145a 太秀了,我把自己蒸馏成了 Skill!已开源 - 程序员鱼皮 ASP.NET Core 内存缓存实战:一篇搞懂该怎么配、怎么避坑 基于 Ghostty 带有分割标签页和为 Claude 编程设计的通知终端 - BugShare AI 焊死入口:教育的“操作系统级”重塑 - 郝hai 初级Java开发工程师使用sql脚本编写代码的过程是简单而且不糊涂 - CoderOilStation Claude Code通关手册(六):MCP协议完全指南 - 暮色之狐 边框灯光环绕动画特效实现指南 - Newbe36524 开源:子木蒸馏版的 SEO 审计工具 seo-audit-skill v1.0 我所理解的Python元模型 【从0到1构建一个ClaudeAgent】规划与协调-TodoWrite - 程序员Seven Claude 和 Codex 在审计 Skill 上性能差异探究 - ACai_sec AScript如何实现中文脚本引擎 - rockey627 【渗透测试】HTB Season10 Garfield 全过程wp - dynasty_chenzi Android 开发者为什么必须掌握 AI 能力?端侧视角下的技术变革 树状数组正确性证明 - AC-wyr 你的 AI 焦虑,可能比 AI 本身更危险——ATM 机没有消灭银行柜员,但恐慌消灭了你的判断力 - 我没有三颗心脏 一个拉胯的分库分表方案有多绝望?整个部门都在救火! - 冰河团队 动态规划入门必学之走方格问题 - Ofnoname PostgREST 与 PostgreSQL 角色权限配置全解析(生产级实践) - SheepDog1998 使用 UEFI 图形输出协议 GOP 在屏幕上显示图像的方法 - 阿源- Claude Code通关手册(五):组建你的AI专家团队,子代理系统 - 暮色之狐 一个程序员到架构师的催婚路之感悟(整整10年后的催婚相亲感悟) - MisterLip 用 Agent Skill 自动生成工作周报 - 赵康
浏览器缓存之【基础键值存储】:Local storage 和 Session storage
橙子家 · 2026-06-17 · via 博客园_首页

〇、前言

虽然在日常口语中我们常把浏览器中 Application -> Storage 模块中的多种类型统称为“浏览器缓存”,但在计算机科学与 Web 开发的专业定义中,实际上包含了“存储”“缓存”两个截然不同但又紧密相关的概念体系。

其中本文将着重介绍的 Local storage 和 Session storage 就属于应用数据存储体系。这部分机制的主要目的是持久化保存数据(如:用户设置、登录状态、业务记录),以便在页面刷新或重启后恢复应用状态。它们通常由 JavaScript 代码主动读写。

LocalStorage 用于长期保存(如:主题偏好),SessionStorage 仅在当前标签页会话有效(如:表单草稿)。它们不是用来加速网页加载的,而是为了记住用户的数据。

下边是详细介绍,供参考。

一、什么是 LocalStorage?

1.1 简介

LocalStorage 是浏览器提供的持久化本地存储方案,属于 Web Storage API 的一部分,并非传统意义上的 HTTP 缓存(如:强缓存、协商缓存)。

主要用于在客户端长期保存非敏感数据,数据在浏览器关闭后仍保留,且不会随 HTTP 请求自动发送至服务器。

核心价值在于减少服务器请求、提升用户体验,适用于用户偏好设置、草稿保存等场景,但需注意容量限制和数据类型约束。

LocalStorage 是轻量级持久化存储的首选方案,适用于 5–10 MB 以内、非敏感的字符串数据,典型场景包括用户偏好、表单草稿和小型缓存。

其核心优势在于免 HTTP 传输开销和跨会话持久性,但需严格规避敏感信息存储,并通过容量监控、数据序列化和精准清理策略规避常见陷阱。

对于超大容量或结构化数据需求,应转向 IndexedDB 等更专业的方案。

1.2 核心特征

1)生命周期与持久性

永久存储:数据除非被手动清除(调用 clear() 或 removeItem())或用户主动清理浏览器缓存,否则长期保留,即使关闭浏览器、重启设备也不会丢失。
同源策略限制:仅限协议、域名、端口完全一致的页面访问(例如 https://example.com 无法读取 http://example.com 的数据)。

2)存储容量与限制

实际容量因浏览器而异
    Chrome/Edge 桌面版:约 10 MB
    Firefox:约 5–10 MB
    Safari(iOS):可能低至 1 MB(内存紧张时更低)
    微信 WebView(X5 内核):常不足 1 MB

注意:早期“5MB 固定容量”的说法已过时,实际空间受设备平台、隐私设置影响。

超限处理:调用 setItem() 时若超出容量,会抛出 QuotaExceededError 异常,不会静默失败或截断数据,需通过 try-catch 捕获并触发清理逻辑。

3)数据类型与操作规范

仅支持字符串存储:直接存入对象/数组会转为 "[object Object]",必须通过 JSON.stringify() 序列化后存储,读取时用 JSON.parse() 还原。
基础操作方法:
    setItem(key, value):存储或覆盖数据。
    getItem(key):读取数据(键不存在时返回 null)。
    removeItem(key):删除指定键。
    clear():清空所有数据。
    key(index) + length:遍历存储项(避免用 for...in 避免原型链污染)。

1.3 典型应用场景

1)用户偏好与配置

保存主题色、语言、字体大小等设置,避免每次加载时重复请求服务器。

// 保存并应用主题
localStorage.setItem('theme', 'dark');
document.body.className = localStorage.getItem('theme');

2)表单草稿自动保存

实时存储输入内容,防止意外关闭页面导致数据丢失:

textarea.addEventListener('input', () => {
  localStorage.setItem('draft', textarea.value);
});
// 页面加载时恢复
window.addEventListener('DOMContentLoaded', () => {
  textarea.value = localStorage.getItem('draft') || '';
});

注意:需通过 JSON.parse()/JSON.stringify() 处理复杂对象。

3)离线数据缓存

存储静态资源(如:配置文件、API 响应快照),在网络中断时提供基础功能支持。

但单条数据超过 100 KB 或总量逼近 5 MB 时,应改用其他缓存类型,如:IndexedDB。

1.4 常见的踩坑点

1)避免存储敏感信息

LocalStorage 无 HttpOnly 保护,若页面存在 XSS 漏洞,攻击者可直接读取数据(如:token)。
敏感信息(如:身份凭证)应优先通过 HttpOnly Cookie 存储,或结合 Web Crypto API 加密后再存。

2)容量超限的预防策略

写入前试探空间:先尝试存入测试数据,成功后立即删除。
预留 10% 余量:按 UTF-8 字节计算真实占用(new TextEncoder().encode(str).length),而非字符串长度。
精准清理机制:按前缀删除非核心数据(如:cache_)、优先清理过期项,避免直接调用 clear()。

3)特殊环境限制

隐身模式:部分浏览器(如:Chrome)会完全禁用 localStorage(调用时抛出 SecurityError)。
移动端 WebView:容量可能显著低于桌面端(如:微信 X5 内核常 <1 MB),需降级处理。

二、什么是 SessionStorage?

2.1 简介

SessionStorage 是浏览器提供的会话级本地存储方案,同样属于 Web Storage API 的一部分,也不同于传统 HTTP 缓存机制(如:强缓存、协商缓存)。

其核心特点是数据仅在当前浏览器标签页的会话周期内有效,标签页关闭后自动清除,且不同标签页的 sessionStorage 相互隔离

它适用于临时数据存储,不会随 HTTP 请求发送至服务器,与服务端的 Session 机制无关。

SessionStorage 是专为单次浏览器会话设计的临时存储方案,核心价值在于标签页内数据的防丢失与状态延续,典型场景包括表单草稿、滚动位置恢复等。其关键优势是自动清理机制(关闭标签页即释放资源),避免数据残留风险。

但需注意:不同标签页数据隔离、隐身模式可能失效,且绝不适用于跨会话或跨标签页的数据共享。若需跨标签页通信,应改用 localStorage + storage 事件方案。

2.2 核心特性

1)生命周期与作用域

会话级时效性:数据仅在当前标签页打开期间有效,即使刷新页面或跳转同源页面仍保留,但关闭标签页后立即销毁。
严格标签页隔离:即使同一域名,不同标签页的 sessionStorage 完全独立,无法跨标签页共享数据(例如:从 example.com 打开新标签页时,新标签页的 sessionStorage 为空)。
同源策略限制:仅限协议、域名、端口完全一致的页面访问(例如:https://a.example.com 无法读取 https://b.example.com 的数据)。

2)存储容量与操作规范

容量限制:与 localStorage 类似,通常为 5–10 MB(实际因浏览器而异),超限时抛出 QuotaExceededError 异常。
仅支持字符串存储:存储对象/数组需通过 JSON.stringify() 序列化,读取时用 JSON.parse() 还原。
基础操作方法:
    setItem(key, value):存储或覆盖数据。
    getItem(key):读取数据(键不存在时返回 null)。
    removeItem(key):删除指定键。
    clear():清空当前标签页的所有数据。

3)事件监听机制

storage 事件的特殊性:当其他标签页修改同源 localStorage 或 sessionStorage 时,会触发当前页面的 storage 事件;但当前页面自身修改 sessionStorage 不会触发该事件。

window.addEventListener('storage', (e) => {
  console.log(`键 ${e.key} 从 ${e.oldValue} 更新为 ${e.newValue}`);
});

此特性可用于跨标签页通信(需结合 localStorage),但 sessionStorage 本身无法直接跨标签页共享数据。

2.3 典型应用场景

1)单次会话的临时数据保护

表单草稿防丢失:用户填写多步骤表单时,将已填内容存入 sessionStorage,即使刷新页面也能恢复进度。

// 保存输入内容
form.addEventListener('input', () => {
  sessionStorage.setItem('formDraft', JSON.stringify(formData));
});
// 页面加载时恢复
window.addEventListener('DOMContentLoaded', () => {
  const draft = sessionStorage.getItem('formDraft');
  if (draft) restoreForm(JSON.parse(draft));
});

2)页面状态临时缓存

滚动位置恢复:从列表页跳转详情页后返回时,自动恢复原滚动位置。

// 列表页:记录退出时的滚动位置
window.addEventListener('beforeunload', () => {
  sessionStorage.setItem('scrollPos', window.scrollY);
});
// 重新进入列表页:恢复位置
window.addEventListener('DOMContentLoaded', () => {
  const pos = sessionStorage.getItem('scrollPos');
  if (pos) window.scrollTo(0, parseInt(pos));
});

3)单次会话的防重复操作

防止表单重复提交:提交后存储标记,避免用户快速多次点击导致重复请求。

submitButton.addEventListener('click', () => {
  if (sessionStorage.getItem('submitting')) return;
  sessionStorage.setItem('submitting', 'true');
  // 执行提交逻辑 ...
});

2.4 常见的踩坑点

1)避免跨标签页依赖

通过 window.open() 或 <a target="_blank"> 打开新标签页时,新标签页的 sessionStorage 为空(部分浏览器可能复制旧数据,但属于非规范行为,不可依赖)

2)隐身/隐私模式的特殊行为

部分浏览器完全禁用:Chrome 等在隐身模式下会抛出 SecurityError(调用 setItem() 时),需通过 try-catch 降级处理4。
Safari 的差异:在隐私浏览模式下,sessionStorage 可写入但关闭标签页后数据立即失效(符合规范)。

3)与“缓存”概念的常见误区

非 HTTP 缓存:SessionStorage 不参与浏览器缓存流程(如:Cache-Control),与资源加载速度无关。
非服务端会话:其名称中的 "session" 仅指浏览器标签页的生命周期,不关联服务端 Session 机制(如:PHP 的 $_SESSION)。

三、两者的关系和区别

LocalStorage 和 SessionStorage 同属 Web Storage API,核心区别在于数据生命周期与作用域:

  • LocalStorage 采用永久存储机制,数据除非手动清除否则长期保留;
  • SessionStorage 采用会话级存储机制,数据仅在当前标签页会话期间有效,关闭标签页后自动清除。

两者均基于同源策略,提供相同的键值对操作接口,且数据不会随 HTTP 请求发送至服务器,避免了 Cookie 的带宽浪费问题。

3.1 LocalStorage 和 SessionStorage 的相同点

基础机制方面的相同点:

同属 Web Storage 规范:两者均为 HTML5 定义的客户端存储方案,用于替代 Cookie 实现更高效的本地数据存储
存储形式一致:均以字符串键值对形式存储数据,复杂对象需通过 JSON.stringify() 和 JSON.parse() 转换。
API 接口相同:均提供 setItem()、getItem()、removeItem()、clear() 和 key() 等方法,操作逻辑完全一致。
同源策略限制:数据仅对相同协议、域名、端口的页面可见,跨源无法访问。

相同的性能优势:

避免网络传输:与 Cookie 不同,存储数据不会随 HTTP 请求自动发送至服务器,显著减少冗余流量。
容量更大:理论存储上限普遍为 5MB~10MB(具体因浏览器而异),远高于 Cookie 的 4KB 限制。
本地快速读取:数据直接从浏览器缓存读取,无需网络请求,提升页面响应速度。

3.2 LocalStorage 和 SessionStorage 的区别

特性 LocalStorage SessionStorage
生命周期 永久存储,除非手动清除或用户主动删除 会话级存储,关闭标签页后自动清除
作用域 同源窗口间全局共享(所有标签页可见) 仅当前标签页有效,不同标签页独立隔离
典型场景 用户偏好设置、持久化缓存、跨页面数据共享 临时表单数据、页面跳转状态、敏感操作流程
关闭浏览器后 数据保留 数据清除
隐私模式 不可读取 不可读取

LocalStorage 和 SessionStorage 通过差异化生命周期设计,分别解决了持久化缓存与临时状态管理的需求。若需数据跨会话留存,选 LocalStorage;若仅需单次会话内有效,选 SessionStorage。两者结合同源策略与高效 API,成为现代 Web 应用客户端存储的核心基础设施。

四、小小的总结

LocalStorage 和 SessionStorage 作为 Web Storage API 的核心成员,共同构成了现代 Web 应用客户端数据存储的基础设施。它们通过标准化的键值对接口,解决了传统 Cookie 在容量与性能上的局限,成为替代 Cookie 实现高效本地存储的首选方案。

  • 核心定位:应用状态管理,而非网络加速

二者本质上属于应用数据存储体系,而非 HTTP 缓存机制。其核心价值在于持久化保存用户数据(如偏好设置、登录状态、业务记录),以便在页面刷新或重启后恢复应用状态,而非加速资源加载。
关键差异:生命周期决定应用场景
  LocalStorage:采用永久存储机制,数据除非手动清除否则长期保留,适用于跨会话的持久化需求(如用户主题偏好、长期缓存)。
  SessionStorage:采用会话级存储机制,数据仅在当前标签页会话期间有效,关闭标签页后自动清除,适用于单次会话的临时状态管理(如表单草稿、页面跳转状态)。

  • 共同约束:同源策略与安全边界

两者均严格遵循同源策略(协议+域名+端口),且数据不会随 HTTP 请求发送至服务器,避免了 Cookie 的带宽浪费问题。

但需注意:
  仅支持字符串存储,复杂对象需序列化。
  容量限制约 5–10 MB(实际因浏览器而异)。
  避免存储敏感信息(如未加密的 token),防范 XSS 攻击风险。

  • 实践指引:按需选择,规避陷阱

选 LocalStorage:当数据需跨会话留存、跨标签页共享时。
选 SessionStorage:当数据仅需单次会话有效、标签页隔离时。
避坑要点:
  - 写入前检测容量余量,捕获 QuotaExceededError。
  - 隐身模式下可能完全禁用,需降级处理。
  - 敏感操作优先使用 HttpOnly Cookie 或服务端会话。

通过合理运用这两种存储机制,开发者可在保障安全的前提下,显著提升 Web 应用的用户体验与响应效率,构建更智能、更可靠的客户端数据管理方案。