浏览器会话存储.标签之间共享?

Vla*_*nko 123 javascript cross-browser session-storage

我在我的网站中有一些值,我希望在浏览器关闭时清除,我选择sessionStorage来存储该值,当选项卡关闭时清除它并在用户按f5时保持存储,但如果用户在不同的选项卡中打开一些链接这个值不可用.我如何在我的应用程序之间共享所有浏览器选项卡之间的sessionStorage值?

用例:将值放在某个存储中,在所有浏览器选项卡中保持该值可访问,如果所有选项卡都关闭则清除它.

if (!sessionStorage.getItem(key)) {
    sessionStorage.setItem(key, defaultValue)
}
Run Code Online (Sandbox Code Playgroud)

naw*_*rgs 112

您可以使用localStorage及其"storage"eventListener将sessionStorage数据从一个选项卡传输到另一个选项卡.

此代码需要存在于所有选项卡上.它应该在您的其他脚本之前执行.

// transfers sessionStorage from one tab to another
var sessionStorage_transfer = function(event) {
  if(!event) { event = window.event; } // ie suq
  if(!event.newValue) return;          // do nothing if no value to work with
  if (event.key == 'getSessionStorage') {
    // another tab asked for the sessionStorage -> send it
    localStorage.setItem('sessionStorage', JSON.stringify(sessionStorage));
    // the other tab should now have it, so we're done with it.
    localStorage.removeItem('sessionStorage'); // <- could do short timeout as well.
  } else if (event.key == 'sessionStorage' && !sessionStorage.length) {
    // another tab sent data <- get it
    var data = JSON.parse(event.newValue);
    for (var key in data) {
      sessionStorage.setItem(key, data[key]);
    }
  }
};

// listen for changes to localStorage
if(window.addEventListener) {
  window.addEventListener("storage", sessionStorage_transfer, false);
} else {
  window.attachEvent("onstorage", sessionStorage_transfer);
};


// Ask other tabs for session storage (this is ONLY to trigger event)
if (!sessionStorage.length) {
  localStorage.setItem('getSessionStorage', 'foobar');
  localStorage.removeItem('getSessionStorage', 'foobar');
};
Run Code Online (Sandbox Code Playgroud)

我用chrome,ff,safari测试了这个,即11,即10,ie9

这个方法"应该在IE8中工作",但我无法测试它,因为我的IE每次打开一个标签时崩溃....任何标签......在任何网站上.(好的IE浏览器)PS:如果你也想要IE8支持,你显然需要包含一个JSON垫片.:)

归功于这篇完整的文章:http: //blog.guya.net/2015/06/12/sharing-sessionstorage-between-tabs-for-secure-multi-tab-authentication/

  • 看起来“BroadcastChannel”现在是执行此操作的好工具。https://developers.google.com/web/updates/2016/09/broadcastchannel (9认同)
  • 不会在之前打开的所有标签中触发事件吗? (4认同)
  • 该解决方案的可靠性如何?由于它是基于事件的,因此是否有错过事件的机会? (2认同)
  • 根据此处的答案,我在localStorage和sessionStorage上创建了一个库以简化此过程。所以现在您只需调用storageManager.saveSyncedSessionData('data','key'); 或storageManager.savePermanentData('data','key');等,根据您的需要。完整的代码在这里:http://www.ebenmonney.com/blog/how-to-implement-remember-me-functionality-using-token-based-authentication-and-localstorage-in-a-web-application (2认同)
  • FWIW,这似乎不再起作用(欢迎,来自未来)。`sessionStorage` 仅在选项卡内跨框架触发“存储”事件,因此这个(非常聪明的)过程将不再适用于跟踪选项卡之间的 sessionStorage 更改。 (2认同)
  • 这不是一个糟糕的解决方案吗,因为所有监听 localStorage 更改的选项卡都可以读取您的 sessionStorage 吗?如果您以这种方式让自己变得脆弱,那么使用 SessionStorage 的全部意义是什么?就像是的,这是可行的,但在某种程度上你只是使用本地存储而不是会话。 (2认同)

Hen*_*son 66

使用sessionStorage此功能是不可能的.

来自MDN Docs

在新选项卡或窗口中打开页面将导致启动新会话.

这意味着你不能在标签之间共享,为此你应该使用 localStorage

  • 好的,我明白了,但是当所有浏览器标签关闭时我怎么能清除localStorage? (4认同)
  • 如果您右键单击 Chrome 中的选项卡并单击“复制”,则“sessionStorage”实际上在原始选项卡和复制选项卡之间共享。 (4认同)
  • @Anmol是的,但我已经不能提供任何代码了,它已经很久了.这是一个丑陋的解决方案,但它完全适合我.关键是将tabsOpened计数器保留在本地存储中,并在页面加载时增加它,但在页面卸载时减少.所以当页面卸载和tabsOpen == 1时它的最后一个标签我们可以清除所有内容.我需要处理一些小事但却无法记住究竟是什么.希望它能帮到你! (2认同)

Ada*_*sko 9

  1. 您可以使用localStorage并记住它首次创建的日期session cookie.当localStorage"会话"早于cookie的值时,您可以清除localStorage

    缺点是有人在浏览器关闭后仍然可以读取数据,因此如果您的数据是私密且有信心的,那么这不是一个好的解决方案.

  2. 您可以将数据存储localStorage几秒钟,并为storage事件添加事件侦听器.通过这种方式,您可以知道任何标签何时写入localStorage内容并且您可以将其内容复制到该标签sessionStorage,然后清除localStorage


Dan*_*ker 8

实际上在查看其他区域时,如果使用_blank打开它,只要在父项打开时打开选项卡,它就会保留sessionStorage:

在这个链接中,有一个很好的jsfiddle来测试它. 当跟踪带有target ="_ blank"的链接时,新窗口上的sessionStorage不为空

  • 从版本 89 开始,这在 Chrome 中不起作用 - https://developer.chrome.com/blog/deps-rems-89/ - 停止为使用 noopener 打开的窗口克隆 sessionStorage (3认同)

Dre*_*oon 8

我发现在选项卡之间共享 sessionStorage 的唯一方法是window.open

  • window.open('./page2.html','')使用新选项卡打开 page2
  • window.open('./page2.html','height=100, width=100')在新窗口中使用新选项卡打开 page2。

Page2可以从page1获取sessionStorage的副本,但这两个sessionStorage对象是相互独立的。