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 相关文章推荐
IE和Firefox在JavaScript应用中的兼容性探讨
Apr 01 Javascript
用JQuery模仿淘宝的图片放大镜显示效果
Sep 15 Javascript
jQuery :first选择器使用介绍
Aug 09 Javascript
jquery用offset()方法获得元素的xy坐标
Sep 06 Javascript
jQuery+css实现的切换图片功能代码
Jan 27 Javascript
纯JS打造网页中checkbox和radio的美化效果
Oct 13 Javascript
JavaScript实现替换字符串中最后一个字符的方法
Mar 07 Javascript
js判断节假日实例代码
Dec 27 Javascript
小程序开发基础之view视图容器
Aug 21 Javascript
Vue瀑布流插件的使用示例
Sep 19 Javascript
微信小程序列表中item左滑删除功能
Nov 07 Javascript
layuiAdmin循环遍历展示商品图片列表的方法
Sep 16 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中in_array函数用法探究
2014/11/25 PHP
PHP解耦的三重境界(浅谈服务容器)
2017/03/13 PHP
Centos7安装swoole扩展操作示例
2020/03/26 PHP
TNC vs BOOM BO3 第二场2.13
2021/03/10 DOTA
javaScript 数值型和字符串型之间的转换
2009/07/25 Javascript
js浮动图片的动态效果
2013/07/10 Javascript
仿百度输入框智能提示的js代码
2013/08/22 Javascript
使用ImageMagick进行图片缩放、合成与裁剪(js+python)
2013/09/16 Javascript
JavaScript中的常见问题解决方法(乱码,IE缓存,代理)
2013/11/28 Javascript
使用JSLint提高JS代码质量方法分享
2013/12/16 Javascript
JavaScript通过prototype给对象定义属性用法实例
2015/03/23 Javascript
jQuery获得包含margin的outerWidth和outerHeight的方法
2015/03/25 Javascript
JS使用oumousemove和oumouseout动态改变图片显示的方法
2015/03/31 Javascript
jquery插件qrcode在线生成二维码
2015/04/26 Javascript
JavaScript生成SQL查询表单的方法
2015/08/13 Javascript
JQuery Ajax 异步操作之动态添加节点功能
2017/05/24 jQuery
深入浅析Vue.js计算属性和侦听器
2018/05/05 Javascript
探索JavaScript中私有成员的相关知识
2019/06/13 Javascript
Vue中多元素过渡特效的解决方案
2020/02/05 Javascript
python解析html开发库pyquery使用方法
2014/02/07 Python
Python实现简单拆分PDF文件的方法
2015/07/30 Python
使用Python爬了4400条淘宝商品数据,竟发现了这些“潜规则”
2018/03/23 Python
将pandas.dataframe的数据写入到文件中的方法
2018/12/07 Python
如何运行.ipynb文件的图文讲解
2019/06/27 Python
Pandas之groupby( )用法笔记小结
2019/07/23 Python
DJANGO-URL反向解析REVERSE实例讲解
2019/10/25 Python
Python实现点云投影到平面显示
2020/01/18 Python
解决import tensorflow as tf 出错的原因
2020/04/16 Python
面向新手解析python Beautiful Soup基本用法
2020/07/11 Python
Python os库常用操作代码汇总
2020/11/03 Python
北美个性化礼品商店:Things Remembered
2018/06/12 全球购物
伦敦一卡通:The London Pass
2018/11/30 全球购物
英国健身超市:Fitness Superstore
2019/06/17 全球购物
美国翻新电子产品商店:The Store
2019/10/08 全球购物
两道JAVA笔试题
2016/09/14 面试题
J2ee常用的设计模式?说明工厂模式
2015/05/21 面试题