localStorage和sessionStorage需按数据生命周期与可见范围严格选用:前者持久同源共享,后者仅限单标签页会话;存对象必用JSON.stringify/parse,禁存敏感信息,storage事件仅响应其他窗口变更。
localStorage 和 sessionStorage 不是“选一个就行”,而是“用错一个,数据就丢或错乱”——关键看数据要活多久、被谁看到。
两者都只接受字符串, se 会自动把非字符串值转成 
[object Object],读出来就是废数据。
JSON.stringify(),比如:localStorage.setItem('user', JSON.stringify({ id: 1, name: 'Alice' }));JSON.parse(),且要加 try...catch 防止损坏的 JSON 崩溃:const user = JSON.parse(localStorage.getItem('user') || '{}');null 或 undefined 会被转成字符串 "null" 或 "undefined",读取时不会自动还原,需自己判断它适合「用户明确希望下次打开还在」的数据,但不是所有长期数据都该放这儿。
theme)、语言设置(lang)、折叠面板状态、搜索历史(非敏感)https://app.example.com 和 https://api.example.com 是两套独立的 localStorage
它的生命周期绑定在「单个标签页」上,不是“一次登录”也不是“一次页面访问”,这点最容易误解。
sessionStorage,和之前完全无关formStep)、OAuth 回调参数暂存、防重复提交标记(submitLock)你在当前窗口改了 localStorage,storage 事件根本不会触发——它只在其他同源窗口修改时通知你。
localStorage.removeItem('token'),B 标签页监听到事件后自动跳转登录页if (typeof localStorage !== 'undefined') { /* 安全使用 */ }真正麻烦的从来不是 API 多难记,而是没想清楚“这个数据到底该不该存在前端”——localStorage 不是数据库,sessionStorage 也不是内存变量,它们只是两个有严格边界的本地抽屉,拿错抽屉,东西就找不到了。