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 相关文章推荐
niceTitle 基于jquery的超链接提示插件
May 31 Javascript
silverlight线程与基于事件驱动javascript引擎(实现轨迹回放功能)
Aug 09 Javascript
JQuery设置获取下拉菜单某个选项的值(比较全)
Aug 05 Javascript
JavaScript异步加载浅析
Dec 28 Javascript
js实现的四级左侧网站分类菜单实例
May 06 Javascript
任意Json转成无序列表的方法示例
Dec 09 Javascript
详解angularjs结合pagination插件实现分页功能
Feb 10 Javascript
React服务端渲染(总结)
Jul 01 Javascript
js实现移动端导航点击自动滑动效果
Jul 18 Javascript
将 vue 生成的 js 上传到七牛的实例
Jul 28 Javascript
es6中使用map简化复杂条件判断操作实例详解
Feb 19 Javascript
js实现特别简单的钟表效果
Sep 14 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获取目标函数执行时间示例
2014/03/04 PHP
PHP 下载文件时如何自动添加bom头及解释BOM头和去掉bom头的方法
2016/01/04 PHP
jQuery对表单的操作代码集合
2011/04/06 Javascript
自己实现ajax封装示例分享
2014/04/01 Javascript
jQuery菜单插件superfish使用指南
2015/04/21 Javascript
原生js页面滚动延迟加载图片
2015/12/20 Javascript
JavaScript实现点击单元格改变背景色的方法
2016/02/12 Javascript
JS+CSS实现的漂亮渐变背景特效代码(6个渐变效果)
2016/03/25 Javascript
BootStrap扔进Django里的方法详解
2016/05/13 Javascript
js实现用户输入的小写字母自动转大写字母的方法
2017/01/21 Javascript
使用 NodeJS+Express 开发服务端的简单介绍
2017/04/07 NodeJs
JavaScript之iterable_动力节点Java学院整理
2017/06/29 Javascript
不使用 JS 匿名函数理由
2017/11/17 Javascript
zTree节点文字过多的处理方法
2017/11/24 Javascript
Angular实现的敏感文字自动过滤与提示功能示例
2017/12/29 Javascript
简单谈谈CommonsChunkPlugin抽取公共模块
2017/12/31 Javascript
vue裁切预览组件功能的实现步骤
2018/05/04 Javascript
Vue函数式组件的应用实例详解
2019/08/30 Javascript
vue-cli4.x创建企业级项目的方法步骤
2020/06/18 Javascript
基于javascript处理nginx请求过程详解
2020/07/07 Javascript
如何通过JS实现日历简单算法
2020/10/14 Javascript
python+pygame简单画板实现代码实例
2017/12/13 Python
python3.5+tesseract+adb实现西瓜视频或头脑王者辅助答题
2018/01/17 Python
windows下 兼容Python2和Python3的解决方法
2018/12/05 Python
Python Matplotlib 基于networkx画关系网络图
2019/07/10 Python
python 计算方位角实例(根据两点的坐标计算)
2020/01/17 Python
基于keras 模型、结构、权重保存的实现
2020/01/24 Python
HTML5 canvas实现雪花飘落特效
2016/03/08 HTML / CSS
生产副总岗位职责
2013/11/28 职场文书
大学生职业生涯规划书汇总
2014/03/20 职场文书
活动总结报告格式
2014/05/09 职场文书
拾金不昧表扬稿
2015/01/16 职场文书
2015年煤矿工作总结
2015/04/28 职场文书
2015年档案管理员工作总结
2015/05/13 职场文书
2016年校园植树节广播稿
2015/12/17 职场文书
Windows下redis下载、redis安装及使用教程
2021/06/02 Redis