jQuery的实现原理的模拟代码 -2 数据部分


Posted in Javascript onAugust 01, 2010

这个数据当然要通过属性来进行存取,但是,有多个属性怎么办呢?,要定义多个属性吗?,属性的名字叫什么呢?会不会与其他的属性有冲突呢?
在 jQuery 中,针对 DOM 对象扩展的私有数据可以用一个对象来表示,多个数据就使用这个对象的多个属性来表示。为了能够通过 DOM 对象找到这个扩展数据对象,而不会与其他现有的属性冲突,在 jQuery 中通过 expando 这个常量表示扩展对象的属性名,这个 expando 的值是计算出来的。而这个属性的值就是用来找到扩展对象的键值。
例如,我们可以定义 expando 的值为 "jQuery1234" ,那么,我们可以为每个 DOM 对象增加这个名为 "jQuery1234" 的属性,这个属性的值可以是一个键,例如为 1000。
在 jQuery 对象上的 cache 用来保存所有对象扩展的对象,这个对象可以看作一个字典,属性名就是键值,所对应的值就是扩展数据对象。
也就是说,在 jQuery 对象的 cache 上,将会有一个 1000 的成员,这个成员引用的对象就是 1000 号 DOM 对象的私有扩展对象。1000 号成员的私有数据将被存在在这个对象上。
当一个 DOM 对象需要取得扩展数据的时候,首先通过对象的 expando 属性取得一个键值,然后通过这个键值到 jQuery.cache 中取得自己的扩展对象,然后在扩展对象上读写数据。

/// <reference path="jQuery-core.js" /> 
// 常用方法 
function now() { 
return (new Date).getTime(); 
} 
// 扩充数据的属性名,动态生成,避免与已有的属性冲突 
var expando = "jQuery" + now(), uuid = 0, windowData = {}; 
jQuery.cache = {}; 
jQuery.expando = expando; 
// 数据管理,可以针对 DOM 对象保存私有的数据,可以读取保存的数据 
jQuery.fn.data = function (key, value) { 
// 读取 
if (value === undefined) { 
return jQuery.data(this[0], key); 
} 
else { // 设置 
this.each( 
function () { 
jQuery.data(this, key, value); 
} 
); 
} 
} 
// 移除数据,删除保存在对象上的数据 
jQuery.fn.removeData = function (key) { 
return this.each(function () { 
jQuery.removeData(this, key); 
}) 
} // 为元素保存数据 
jQuery.data = function (elem, name, data) { // #1001 
// 取得元素保存数据的键值 
var id = elem[expando], cache = jQuery.cache, thisCache; 
// 没有 id 的情况下,无法取值 
if (!id && typeof name === "string" && data === undefined) { 
return null; 
} 
// Compute a unique ID for the element 
// 为元素计算一个唯一的键值 
if (!id) { 
id = ++uuid; 
} 
// 如果没有保存过 
if (!cache[id]) { 
elem[expando] = id; // 在元素上保存键值 
cache[id] = {}; // 在 cache 上创建一个对象保存元素对应的值 
} 
// 取得此元素的数据对象 
thisCache = cache[id]; 
// Prevent overriding the named cache with undefined values 
// 保存值 
if (data !== undefined) { 
thisCache[name] = data; 
} 
// 返回对应的值 
return typeof name === "string" ? thisCache[name] : thisCache; 
} 
// 删除保存的数据 
jQuery.removeData = function (elem, name) { // #1042 
var id = elem[expando], cache = jQuery.cache, thisCache = cache[id]; 
// If we want to remove a specific section of the element's data 
if (name) { 
if (thisCache) { 
// Remove the section of cache data 
delete thisCache[name]; 
// If we've removed all the data, remove the element's cache 
if (jQuery.isEmptyObject(thisCache)) { 
jQuery.removeData(elem); 
} 
} 
// Otherwise, we want to remove all of the element's data 
} else { 
delete elem[jQuery.expando]; 
// Completely remove the data cache 
delete cache[id]; 
} 
} 
// 检查对象是否是空的 
jQuery.isEmptyObject = function (obj) { 
// 遍历元素的属性,只有要属性就返回假,否则返回真 
for (var name in obj) { 
return false; 
} 
return true; 
} 
// 检查是否是一个函数 
jQuery.isFunction = function (obj) { 
var s = toString.call(obj); 
return toString.call(obj) === "[object Function]"; 
}

下面的脚本可以保存或者读取对象的扩展数据。
// 数据操作 
$("#msg").data("name", "Hello, world."); 
alert($("#msg").data("name")); 
$("#msg").removeData("name"); 
alert($("#msg").data("name"));
Javascript 相关文章推荐
JavaScript 在各个浏览器中执行的耐性
Apr 06 Javascript
javascript 清除输入框中的数据
Apr 13 Javascript
javascript Array数组对象的扩展函数代码
May 22 Javascript
jquery mobile实现拨打电话功能的几种方法
Aug 05 Javascript
jQuery表格插件ParamQuery简单使用方法示例
Dec 05 Javascript
jqGrid读取选择的多行的某个属性代码
May 18 Javascript
JS遍历页面所有对象属性及实现方法
Aug 01 Javascript
jQuery.form.js的使用详解
Jun 14 jQuery
JavaScript使用FileReader实现图片上传预览效果
Mar 27 Javascript
使用vue的v-for生成table并给table加上序号的实例代码
Oct 27 Javascript
详解在Vue.js编写更好的v-for循环的6种技巧
Apr 14 Javascript
详解TS数字分隔符和更严格的类属性检查
May 06 Javascript
jQuery的实现原理的模拟代码 -1 核心部分
Aug 01 #Javascript
jQuery validate 中文API 附validate.js中文api手册
Jul 31 #Javascript
jQuery对象[0]是什么含义?
Jul 31 #Javascript
动态调用CSS文件的JS代码
Jul 29 #Javascript
date.parse在IE和FF中的区别
Jul 29 #Javascript
iframe自适应宽度、高度 ie6 7 8,firefox 3.86下测试通过
Jul 29 #Javascript
jquery中对表单的基本操作代码
Jul 29 #Javascript
You might like
PHP图片处理之使用imagecopyresampled函数裁剪图片例子
2014/11/19 PHP
php简单的上传类分享
2016/05/15 PHP
php使用yield对性能提升的测试实例分析
2019/09/19 PHP
tp5 sum某个字段相加得到总数的例子
2019/10/18 PHP
对字符串进行HTML编码和解码的JavaScript函数
2010/02/01 Javascript
js 文件引入实现代码
2010/04/23 Javascript
javascript五图轮播切换实用版
2012/08/17 Javascript
JS 删除字符串最后一个字符的实现代码
2014/02/20 Javascript
网页右下角弹出窗体实现代码
2014/06/05 Javascript
js数组的基本操作(很全自己整理的)
2014/10/16 Javascript
浅谈javascript 函数内部属性
2015/01/21 Javascript
jQuery使用prepend()方法在元素前添加内容用法实例
2015/03/26 Javascript
js实现左侧网页tab滑动门效果代码
2015/09/06 Javascript
14 个折磨人的 JavaScript 面试题
2016/08/08 Javascript
nodeJS实现简单网页爬虫功能的实例(分享)
2017/06/08 NodeJs
jQuery除指定区域外点击任何地方隐藏DIV功能
2017/11/13 jQuery
学习使用ExpressJS 4.0中的新Router的用法
2018/11/06 Javascript
Vue+Vant 图片上传加显示的案例
2020/11/03 Javascript
浅谈Ant Design Pro 菜单自定义 icon
2020/11/17 Javascript
[05:02]2014DOTA2 TI中国区预选赛精彩TOPPLAY第三弹
2014/06/25 DOTA
二种python发送邮件实例讲解(python发邮件附件可以使用email模块实现)
2013/12/03 Python
python编程开发之类型转换convert实例分析
2015/11/13 Python
python实现QQ邮箱/163邮箱的邮件发送
2019/01/22 Python
python实现引用其他路径包里面的模块
2020/03/09 Python
python 使用多线程创建一个Buffer缓存器的实现思路
2020/07/02 Python
使用CSS3来代替JS实现交互
2017/08/10 HTML / CSS
Mountain Warehouse澳大利亚官网:欧洲家庭户外品牌倡导者
2016/11/20 全球购物
村级个人对照检查材料
2014/08/22 职场文书
开展批评与自我批评发言稿
2014/10/16 职场文书
2014年销售工作总结
2014/12/01 职场文书
2015年毕业生实习评语
2015/03/25 职场文书
2015年企业团支部工作总结
2015/05/21 职场文书
培训后的感想
2015/08/07 职场文书
企业廉洁教育心得体会
2016/01/20 职场文书
python字符串拼接.join()和拆分.split()详解
2021/11/23 Python
PostgreSQL数据库去除重复数据和运算符的基本查询操作
2022/04/12 PostgreSQL