HTML 5本地存储之兼容性与存储监听

开发 前端
onstorage可用时,监听事件并在事件触发时判断是否指定的key,onstorage不可用(IE8以下、Chrome因domain问题)时使用Timer来检查。

很早之前调研过HTML5的本地存储-《DOM Storage全解析》,大致上对localStorage、sessionStorage等API做了下了解,但是一直没有机会真正的在项目中使用。终于这次借重构Web IM的机会,对本地存储做了更深入的使用,除了基本的API使用之外还在次基础上封装了一些应用层的库,例如Tab之间的操作同步、Tab之间的请求同步等。本文主要做一个阶段性的经验总结。

一、localStorge onstorage事件的兼容性

1. 触发情况

IE8/IE9/Firefox3.6: 在页面A中注册onstorage事件,修改localStorage时,A页面和其他页面都能收到onstorage事件。因此,对于这些浏览器监听onstorage时需要自己判断是否是本页面触发的,并且忽略本页面触发的行为。

Chrome12/Firefox4/Opera11/Safari5中只有收到由其他页面触发的onstorage事件。

此外,Chrome14 DEV版本中测试发现,在页面设置了document.domain之后,onstorage事件无论如何都不会触发,此Bug导致在Chrome下无法使用onstorage事件。

2. 事件注册

IE需要注册在document上,其他均注册在window上。

JavaScript Code复制内容到剪贴板
  1. //IE注册在document上       
  2. if( document.attachEvent && !K.Browser.opera ) {       
  3.     document.attachEvent("onstorage", _onstorage(key,callback));       
  4. }       
  5. //其他注册在window上       
  6. else{       
  7.     window.addEventListener("storage", _onstorage(key,callback), false);       
  8. };       
 

3. 事件对象

IE中的storageEvent对象不包含key/newValue/oldValue等属性,因此如果想知道是哪个Key的数据发生了变化需要自己处理,其他浏览器则可以直接获得数据。

4. 数据的获取

IE9下在事件触发时尽然无法立即获取到对应key的值,需要使用setTimeout做异步处理。其他浏览器状况良好。

二、监听某个Key的变化

监听某个key也就是在onstorage的基础上更精细一些,这是之后各种应用的基础。以下为实现方案:

1. onstorage可用时,监听事件并在事件触发时判断是否指定的key

2. onstorage不可用(IE8以下、Chrome因domain问题)时使用Timer来检查

JavaScript Code复制内容到剪贴板

  1. var LocalStorage = (function(){       
  2.         var ls = window.localStorage;       
  3.        
  4.         function _onstorage( key, callback ){       
  5.             var oldValue = ls[key];       
  6.             /*     
  7.                 IE下即使是当前页面触发的数据变更,当前页面也能收到onstorage事件,其他浏览器则只会在其他页面收到     
  8.              */       
  9.             return function( e ){       
  10.                 //IE下不使用setTimeout尽然获取不到改变后的值?!       
  11.         setTimeout( function(){       
  12.             e = e || window.storageEvent;       
  13.        
  14.             var tKey = e.key,       
  15.                 newValue = e.newValue;       
  16.             //IE下不支持key属性,因此需要根据storage中的数据判断key中的数据是否变化       
  17.             if( !tKey ){       
  18.                 var nv = ls[key];       
  19.                 if( nv != oldValue ){       
  20.                     tKey = key;       
  21.                     newValue = nv;       
  22.                 }       
  23.        
  24.             }       
  25.        
  26.             if( tKey == key ){       
  27.                 callback && callback(newValue);       
  28.        
  29.                 oldValue = newValue;       
  30.             }       
  31.         }, 0 );       
  32.             }       
  33.         }       
  34.     return {       
  35.         getItem: function( key ){       
  36.             return ls.getItem( key );       
  37.         },       
  38.         setItem: function( key, val ){       
  39.             return ls.setItem( key, val );       
  40.         },       
  41.         removeItem: function( key, val ){       
  42.             return ls.removeItem( key );       
  43.         },       
  44.         clear: function(){       
  45.             return ls.clear();       
  46.         },       
  47.         onstorage: function( key, callback ){       
  48.             //IE6/IE7/Chrome使用Timer检查更新,其他使用onstorage事件       
  49.             /*     
  50.                 Chrome下(14.0.794.0)重写了document.domain之后会导致onstorage不触发     
  51.                 鉴于onstorage的兼容性问题暂时不使用onstorage事件,改用传统的轮询方式检查数据变化     
  52.             */       
  53.             var b = K.Browser;       
  54.        
  55.             if( !this.useTimer ){       
  56.                 //IE注册在document上       
  57.                 if( document.attachEvent && !K.Browser.opera ) {       
  58.                     document.attachEvent("onstorage", _onstorage(key,callback));       
  59.                 }       
  60.                 //其他注册在window上       
  61.                 else{       
  62.                     window.addEventListener("storage", _onstorage(key,callback), false);       
  63.                 };       
  64.             }       
  65.             else{       
  66.                 /*     
  67.                     Timer检查方式     
  68.                  */       
  69.                 var listener = _onstorage( key, callback );       
  70.                 setInterval(function(){       
  71.                     listener({});       
  72.                 }, this.interval);       
  73.             }       
  74.         },       
  75.         //是否使用Timer来check       
  76.         useTimer: ( K.Browser.ie && K.Browser.ie < 8 ) || ( K.Browser.chrome ),       
  77.         //检查storage是否发生变化的时间间隔       
  78.         interval: 1000       
  79.     };       
  80. })();       
以上是LocalStorage接口的完整封装,在localStorage不可用时使用UserData等其他替代方案来实现以上的接口即可。
  1. HTML 5拖拽预览
  2. HTML 5开发:地理位置定位指南
  3. 如何在ASP.NET网站中使用HTML 5拖放功能
  4. 关于HTML 5几个重要安全问题
  5. 教你用CSS3打造HTML5的Logo
责任编辑:张伟 来源: HTML5China
相关推荐

2015-08-13 15:56:44

HTML5本地存储Localstorag

2013-06-21 10:33:02

虚拟化应用存储虚拟化

2017-01-03 18:09:33

HTML5本地存储Web

2015-05-22 10:06:58

2011-03-16 09:49:54

HTML 5

2011-05-11 18:00:10

HTML5兼容性

2011-06-07 15:14:09

HTML 5

2009-06-09 09:30:05

Linux上网本兼容性

2023-04-17 19:43:54

兼容性测试软件测试

2012-04-12 16:32:46

Lumia 800HTML 5兼容性

2013-11-25 22:04:31

华为存储Windows Ser

2009-08-17 10:22:19

C# Windows

2009-03-07 09:49:07

Windows 7兼容性

2017-07-28 08:07:05

2011-05-19 14:02:30

笔记本内存

2023-03-24 07:31:58

Oracle兼容性产品

2011-08-16 15:17:44

IOS SDK

2021-12-27 16:22:19

鸿蒙HarmonyOS应用

2010-03-05 17:09:18

2010-08-19 09:59:03

Office 2011兼容性
点赞
收藏

51CTO技术栈公众号