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 数组的方法集合
Jun 05 Javascript
jQuery Ajax 全解析
Feb 08 Javascript
Javascript面向对象编程(二) 构造函数的继承
Aug 28 Javascript
Javascript实现带关闭按钮的网页漂浮广告代码
Jan 12 Javascript
纯js写的分页表格数据为json串
Feb 18 Javascript
Spring MVC中Ajax实现二级联动的简单实例
Jul 06 Javascript
javascript cookie基础应用之记录用户名的方法
Sep 20 Javascript
正则 js分转元带千分符号详解
Mar 08 Javascript
利用ES6的Promise.all实现至少请求多长时间的实例
Aug 28 Javascript
react redux入门示例
Apr 19 Javascript
解决vue admin element noCache设置无效的问题
Nov 12 Javascript
js实现点击按钮随机生成背景颜色
Sep 05 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基础知识:函数基础知识
2006/12/13 PHP
php完全过滤HTML,JS,CSS等标签
2009/01/16 PHP
探讨PHP中OO之静态关键字以及类常量的详解
2013/06/07 PHP
PHP数字字符串左侧补0、字符串填充和自动补齐的几种方法
2014/05/10 PHP
php抽象方法和抽象类实例分析
2016/12/07 PHP
PHP 表单提交及处理表单数据详解及实例
2016/12/27 PHP
thinkphp实现把数据库中的列的值存到下拉框中的方法
2017/01/20 PHP
通过PHP设置BugFree获取邮箱通知
2019/04/25 PHP
php写入文件不覆盖的实例讲解
2019/09/17 PHP
jQuery获取浏览器类型和版本号的方法
2016/07/05 Javascript
Javascript基于jQuery UI实现选中区域拖拽效果
2016/11/25 Javascript
jQuery的ajax中使用FormData实现页面无刷新上传功能
2017/01/16 Javascript
Vue.js实现微信过渡动画左右切换效果
2017/06/13 Javascript
[js高手之路]从原型链开始图解继承到组合继承的产生详解
2017/08/28 Javascript
Vue多种方法实现表头和首列固定的示例代码
2018/02/02 Javascript
基于vue.js实现的分页
2018/03/13 Javascript
JavaScript实现的前端AES加密解密功能【基于CryptoJS】
2018/08/28 Javascript
关于单文件组件.vue的使用
2018/09/20 Javascript
JS数据类型(基本数据类型、引用数据类型)及堆和栈的区别分析
2020/03/04 Javascript
Python Web框架Flask中使用百度云存储BCS实例
2015/02/08 Python
Python脚本实现网卡流量监控
2015/02/14 Python
解决项目pycharm能运行,在终端却无法运行的问题
2019/01/19 Python
python实现nao机器人身体躯干和腿部动作操作
2019/04/29 Python
将python文件打包成EXE应用程序的方法
2019/05/22 Python
python3射线法判断点是否在多边形内
2019/06/28 Python
Python用字典构建多级菜单功能
2019/07/11 Python
python中68个内置函数的总结与介绍
2020/02/24 Python
Python将字典转换为XML的方法
2020/08/01 Python
html5 css3实例教程 一款html5和css3实现的小机器人走路动画
2014/10/20 HTML / CSS
HTML 5.1来了 9月份正式发布 更新内容预览
2016/04/26 HTML / CSS
一年级班主任寄语
2014/01/19 职场文书
客户答谢会活动方案
2014/08/31 职场文书
疾病证明书
2015/06/19 职场文书
团拜会主持词
2015/07/04 职场文书
2015年物业公司保洁工作总结
2015/10/22 职场文书
PHP判断是否是json字符串
2021/04/01 PHP