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 相关文章推荐
为Extjs加加速(javascript加速)
Aug 19 Javascript
javascript各浏览器中option元素的表现差异
Apr 07 Javascript
基于jquery中children()与find()的区别介绍
Apr 26 Javascript
js操纵跨frame的三级联动select下拉选项实例介绍
May 19 Javascript
JavaScript移除数组元素减少长度的方法
Sep 05 Javascript
JS+CSS实现大气清新的滑动菜单效果代码
Oct 22 Javascript
Bootstrap按钮组件详解
Apr 26 Javascript
js对象浅拷贝和深拷贝详解
Sep 05 Javascript
最实用的jQuery分页插件
Oct 09 Javascript
详解webpack与SPA实践之开发环境搭建
Dec 18 Javascript
react的滑动图片验证码组件的示例代码
Feb 27 Javascript
JavaScript创建表格的方法
Apr 13 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(视频)Http下载
2006/12/12 PHP
一些被忽视的PHP函数(简单整理)
2010/04/30 PHP
PHP5.3与5.5废弃与过期函数整理汇总
2014/07/10 PHP
php数组实现根据某个键值将相同键值合并生成新二维数组的方法
2017/04/26 PHP
ThinkPHP5.1框架数据库链接和增删改查操作示例
2019/08/03 PHP
JS option location 页面跳转实现代码
2008/12/27 Javascript
需要做特殊处理的DOM元素属性的访问
2010/11/05 Javascript
js中的值类型和引用类型小结 文字说明与实例
2010/12/12 Javascript
一行代码实现纯数据json对象的深度克隆实现思路
2013/01/09 Javascript
js或者jquery判断图片是否加载完成实现代码
2013/03/20 Javascript
js中indexof的用法详细解析
2013/12/24 Javascript
JavaScript打印网页指定区域的例子
2014/05/03 Javascript
从JQuery源码分析JavaScript函数的apply方法与call方法
2014/09/25 Javascript
下一代Bootstrap的5个特点 超酷炫!
2016/06/17 Javascript
利用HTML5+Socket.io实现摇一摇控制PC端歌曲切换
2017/01/13 Javascript
node.js基于fs模块对系统文件及目录进行读写操作的方法详解
2017/11/10 Javascript
详解JavaScript的BUG和错误
2018/05/07 Javascript
vue中组件的过渡动画及实现代码
2018/11/21 Javascript
Next.js项目实战踩坑指南(笔记)
2018/11/29 Javascript
ES6知识点整理之模块化的应用详解
2019/04/15 Javascript
原生javascript制作的拼图游戏实现方法详解
2020/02/23 Javascript
python中threading超线程用法实例分析
2015/05/16 Python
numpy.meshgrid()理解(小结)
2019/08/01 Python
python文字转语音实现过程解析
2019/11/12 Python
Django 创建后台,配置sqlite3教程
2019/11/18 Python
Python通过kerberos安全认证操作kafka方式
2020/06/06 Python
Python爬虫设置ip代理过程解析
2020/07/20 Python
CSS3,线性渐变(linear-gradient)的使用总结
2017/01/09 HTML / CSS
英国最大的老式糖果店:A Quarter Of
2017/04/08 全球购物
德国孕妇装和婴童服装网上商店:bellybutton
2018/04/12 全球购物
Otiumberg官网:英国半精致珠宝品牌
2021/01/16 全球购物
枚举与#define宏的区别
2014/04/30 面试题
综治维稳工作承诺书
2014/08/30 职场文书
学校运动会广播稿100条
2014/09/14 职场文书
人代会简报
2015/07/21 职场文书
vue+elementui 实现新增和修改共用一个弹框的完整代码
2021/06/08 Vue.js