读jQuery之六 缓存数据功能介绍


Posted in Javascript onJune 21, 2011

很多同学在项目中都喜欢将数据存储在HTMLElement属性上,如

<div data="some data">Test</div> 
<script> 
div.getAttribute('data'); // some data 
</script>

给页面中div添加了自定义属性“data”及值“some data”。后续JS代码中使用getAttribute获取。
jQuery从1.2.3开始提供了data/removeData方法用来存储/删除数据。1.6.1代码片段
jQuery.extend({ 
cache: {}, 
// Please use with caution 
uuid: 0, 
... 
});

即给jQuery添加了静态字段/方法,有jQuery.cache/jQuery.uuid/jQuery.expando等。下面分别介绍
jQuery.cache 空对象,用来缓存。它的结构较复杂。
jQuery.uuid 自增唯一的数字。
jQuery.expando 字符串,使用Math.random生成,去掉了非数字字符。它作为HTMLElement或JS对象的属性名。
expando: "jQuery" + ( jQuery.fn.jquery + Math.random() ).replace( /\D/g, "" ),

jQuery.noData JS对象,对于指定的HTMLElement禁用data方法。如embed、applet。
jQuery.hasData 用来判断HTMLElement或JS对象是否具有数据。返回true或false。即如果调用了jQuery.data方法添加了属性,则返回true。
<div>aa</div> 
<script> 
var div = document.getElementsByTagName('div')[0]; 
$.hasData(div); // false 
$.data(div, 'name','jack'); 
$.hasData(div); // true 
</script>

jQuery.acceptData 用来判断该元素是否能接受数据,返回true或false。在jQuery.data中使用。
jQuery.data 这是提供给客户端程序员使用的方法,它同时是setter/getter。
1,传一个参数,返回附加在指定元素的所有数据,即thisCachejQuery.data(el); // thisCache
2,传二个参数,返回指定的属性值jQuery.data(el, 'name');
3,传三个参数,设置属性及属性值jQuery.data(el, 'name', 'jack');jQuery.data(el, 'uu', {});
4,传四个参数,第四个参数pvt仅提供给jQuery库自身使用。即jQuery._data方法中传true。因为jQuery的事件模块严重依赖于jQuery.data,为避免人为的不小心重写在这个版本中加入的。
jQuery.removeData 删除数据。
上面是jQuery数据缓存模块的整体概述,下面详细说下jQuery.data方法。jQuery.data为两种对象提供缓存:JS对象和HTMLElement
// 为JS对象提供缓存 
var myObj = {}; 
$.data(myObj, 'name', 'jack'); 
$.data(myObj, 'name'); // jack 
// 为HTMLElement提供缓存 
<div id="xx"></div> 
<script> 
var el = document.getElementById('xx'); 
$.data(el, 'name', 'jack'); 
$.data(el, 'name'); // jack 
</script>

内部实现上也是有区别的,
1,为JS对象提供缓存时,直接将数据保存在JS对象上。cache为JS对象。此时会偷偷的给JS对象添加个属性(类似于jQuery16101803968874529044),属性值也是个JS对象。举例说明
var myObj = {}; 
$.data(myObj, 'name', 'jack'); 
console.log(myObj);

myObj的结构如下
myObj = { 
jQuery16101803968874529044 : { 
name : 'jack' 
} 
}

“jQuery16101803968874529044”这个字符串在data内部命名为id(注意并非HTMLElement元素的id),它实际就是jQuery.expando。上面已经提到它是在jQuery.js引入到页面后随机生成的。
2,为HTMLElement提供缓存时,却不会直接保存在HTMLElement上。而是保存在jQuery.cache上。cache为jQuery.cache。此时先给HTMLElement添加属性(类似于jQuery16101803968874529044),属性值为数字(1,2,3递增)。即只将一些数字保存在了HTMLElement上,不会直接将数据置入。这是因为IE老版本中可能会存在内存泄露危险。而HTMLElement如何与jQuery.cache建立联系呢? 还是id。刚刚提到属性值数字就是id。举例说明
<div id="xx"></div> 
<script> 
var el = document.getElementById('xx'); 
$.data(el, 'name', 'jack'); 
console.log(el[jQuery.expando]); // 1 
console.log(jQuery.cache); // {1 : {name:'jack'}} 
</script>

el 上添加了属性jQuery.expando,值为id,这个id是从1开始递增的。而id又作为jQuery.cache的属性(key)。这样就HTMLElement就与jQuery.cache建立了联系。如图

读jQuery之六 缓存数据功能介绍

不知注意到没有,jQuery.data还有第四个参数pvt,这个参数只在jQuery._data中使用。

// For internal use only. 
_data: function( elem, name, data ) { 
return jQuery.data( elem, name, data, true ); 
},

jQuery._data从命名上就指定它是私有的,使用jQuery的客户端程序员不应该去调用该方法。jQuery的API文档上也不会公开它。
jQuery的数据缓存模块从1.2.3到1.6.1几乎每个版本都在变。jQuery._data的提出就是为了避免客户端程序员覆盖/重写了默写模块。如jQuery事件模块中事件handler等就使用jQuery.data存储,如果重写了该模块。那么事件模块将瘫痪。因此特意添加了pvt参数及jQuery._data方法。
但如果你刻意要破坏,那么还是可以做的。如下
<div id="xx">Test</div> 
<script> 
$('#xx').click(function(){ 
alert('click'); 
}); 
// 语句1 
$.data($('#xx')[0], 'events', '', true); 
// 语句2 
//$._data($('#xx')[0], 'events', ''); 
</script>

点击div[id=xx]将不会触发点击事件。
整个jQuery.data设置(set)数据缓存的过程就是如此,理解的这个。取数据(get)的过程就好理解了。不重复。
最后,我会给zChian.js添加zChain.data/removeData方法,因为是“迷你版”,仅给HTMLElement添加数据缓存。请注意。

相关:
http://msdn.microsoft.com/en-us/library/Bb250448
http://bugs.jquery.com/ticket/6807
zChain-0.6.js

Javascript 相关文章推荐
Javascript代码混淆综合解决方案-Javascript在线混淆器
Dec 18 Javascript
基于jquery的无刷新分页技术
Jun 11 Javascript
第二次聊一聊JS require.js模块化工具的基础知识
Apr 17 Javascript
angular实现商品筛选功能
Feb 01 Javascript
MvcPager分页控件 适用于Bootstrap
Jun 03 Javascript
js插件实现图片滑动验证码
Sep 29 Javascript
jQuery实现点击DIV同时点击CheckBox,并为DIV上背景色的实例
Dec 18 jQuery
js防抖和节流的深入讲解
Dec 06 Javascript
JS实现获取数组中最大值或最小值功能示例
Mar 02 Javascript
微信小程序中使用Async-await方法异步请求变为同步请求方法
Mar 28 Javascript
微信小程序学习总结(五)常见问题实例小结
Jun 04 Javascript
原生js实现照片墙效果
Oct 13 Javascript
将HTMLCollection/NodeList/伪数组转换成数组的实现方法
Jun 20 #Javascript
读jQuery之五(取DOM元素)
Jun 20 #Javascript
读jQuery之四(优雅的迭代)
Jun 20 #Javascript
火狐4、谷歌12不支持Jquery Validator的解决方法分享
Jun 20 #Javascript
合并table相同单元格的jquery插件分享(很精简)
Jun 20 #Javascript
functional继承模式 摘自javascript:the good parts
Jun 20 #Javascript
jQuery数组处理方法汇总
Jun 20 #Javascript
You might like
php.ini修改php上传文件大小限制的方法详解
2013/06/17 PHP
Laravel 4 初级教程之Pages、表单验证
2014/10/30 PHP
PHP处理CSV表格文件的常用操作方法总结
2016/07/01 PHP
PDO::commit讲解
2019/01/27 PHP
Laravel框架中队列和工作(Queues、Jobs)操作实例详解
2020/04/06 PHP
广告代码静态化js通用函数
2007/05/09 Javascript
extjs 为某个事件设置拦截器
2010/01/15 Javascript
关于js内存泄露的一个好例子
2013/12/09 Javascript
js实现select选择框效果及美化
2016/08/19 Javascript
URL的参数中有加号传值变为空格的问题(URL特殊字符)
2016/11/04 Javascript
Javascript 数组去重的方法(四种)详解及实例代码
2016/11/24 Javascript
Vue自定义过滤器格式化数字三位加一逗号实现代码
2018/03/23 Javascript
基于vue写一个全局Message组件的实现
2019/08/15 Javascript
jquery实现吸顶导航效果
2020/01/08 jQuery
ES2020 新特性(种草)
2020/01/12 Javascript
[46:16]2018DOTA2亚洲邀请赛3月30日 小组赛B组 iG VS VP
2018/03/31 DOTA
使用Python来编写HTTP服务器的超级指南
2016/02/18 Python
python奇偶行分开存储实现代码
2018/03/19 Python
PyTorch学习笔记之回归实战
2018/05/28 Python
使用Python的Turtle库绘制森林的实例
2019/12/18 Python
TensorFlow实现保存训练模型为pd文件并恢复
2020/02/06 Python
Python 线性回归分析以及评价指标详解
2020/04/02 Python
python百行代码自制电脑端网速悬浮窗的实现
2020/05/12 Python
Python使用内置函数setattr设置对象的属性值
2020/10/16 Python
python 实现图片裁剪小工具
2021/02/02 Python
CSS3之背景尺寸Background-size使用介绍
2013/10/14 HTML / CSS
canvas简单连线动画的实现代码
2020/02/04 HTML / CSS
佐卡伊官网:中国知名珠宝品牌
2017/02/05 全球购物
奥地利网上现代灯具和灯饰店:Lampenwelt.at
2018/01/29 全球购物
全国法院系统开展党的群众路线教育实践活动综述(全文)
2014/10/25 职场文书
2014小学数学教研组工作总结
2014/12/06 职场文书
大学生考试作弊被抓检讨书
2014/12/27 职场文书
大明湖导游词
2015/02/03 职场文书
大学生求职信怎么写
2015/03/19 职场文书
2016年学校禁毒宣传活动工作总结
2016/04/05 职场文书
python爬虫之利用selenium模块自动登录CSDN
2021/04/22 Python