jQuery 源码分析笔记(6) jQuery.data - Web前端
作者:98wpeu发布时间:2026-06-21分类:网页前端技术浏览:3
导读:data部分的代码从1381行开始。最开始的几行关键代码:复制代码代码如下:JQuery.extend({//存储数据的地方,关键实现核心CAChe:{},//分...
data部分的代码从1381行开始。最开始的几行关键代码:
复制代码 代码如下:
JQuery.extend({
// 存储数据的地方,关键实现核心
CAChe: { },
// 分配ID用的seed
uUId: 0,
// 为了区别不同的jQuery实例存储的数据,使用前缀+jquery版本号+随机数作为Key
expando: "jquery" + (JQuery.fn.jquery + Math.ranDOM()).Replace(/\D/g, ""),
// 以下元素没有Data:embed和applet(这玩意还活着么),除了FLASH之外的Object。
noData: {
"embed": true,
"object": "clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",
"Applet": true
}
});
对外的接口都调用了两个内部函数:jQuery.data(elem, name, data, pvt)和jQuery.removeData(elem, name, pvt)。而removeData的逻辑与data类似,只是data是加入数据,而removeData使用delete或者设置为null删除数据。
data部分的代码中明确区分了js对象和dom对象的保存,这是为了解决部分浏览器的内存泄漏问题。在低版本ie中,当DOM和JS对象之间出现循环引用时,GC就无法正确处理。参见Understanding and Solving Internet Explorer Leak Patterns。至于COM对象,因为已经限制object元素没有data,就绕过了这个问题。
复制代码 代码如下:
data: function(elem, name, data, pvt) {
// 如果属于noData中定义的元素
if(!jQuery.acceptData(elem)) {
return;
}
varinternalKey = jQuery.expando,
getByName = typeof name === "string",
thisCache,
isnode = elem.nodeType,
// DOM元素需要保存在Cache,js对象直接保存到elem
cache = isNode ? jQuery.cache : elem,
// 如果elem的jQuery.expando已经有值了,就重用
id = isNode ? elem[jQuery.expando] : elem[jQuery.expando] && jQuery.expando;
<PRE class=brush:;gutter:true;><Code>// data未定义,说明当前调用是查询数据,但是对象没有任何数据,直接返回
if((!id || (pvt && id && !cache[id][internalKey])) && getByName && data === undefined) {
return;
}
if(!id) {
if(isNode) {
// 用uuid种子递增分配唯一ID,只有DOM元素需要。因为需要存在全局cache中
elem[jQuery.expando] = id = ++jQuery.uuid;
} else {
id = jQuery.expando;
}
}
// 清空原来的值
if(!cache[id]) {
cache[id] = {};
if(!isNode) {
cache[id].tojson = jQuery.noop;
}
}
// 用extend扩展cache,增加一个属性,用来保存数据
if(typeof name === "object" || typeof name === "function") {
if(pvt) {
cache[id][internalKey] = jQuery.expand(cache[id][internalKey], name);
} else {
cache[id] = jQuery.extend(cache[id], name);
}
}
thisCache = cahce[id];
// 避免Key冲突
if(pvt) {
if(!thisCache[internalKey]) {
thisCahce[internalKey] = {};
}
thisCache = thisCache[internalKey];
}
if(data !== undefined) {
thisCache[jQuery.camelCase(name)] = data;
}
return getByName ? thisCache[jQuery.camelCase(name)] : thisCache;
}
removeData: function( elem, name, pvt ) { // 前面部分与data类似 // ... // 部分浏览器不支持在Element上进行delete操作,在jQuery.support中检查过这个浏览器特性。 // 如果delete失败的话,就先设置成Null。 if ( jQuery.support.deleteExpando || cache != window ) { delete cache[ id ]; } else { cache[ id ] = null; }
<PRE class=brush:;gutter:true;><code>var internalCache = cache[ id ][ internalKey ];
// 如果还有数据,就清空一次再设置,增加性能
if ( internalCache ) {
cache[ id ] = {};
cache[ id ][ internalKey ] = internalCache;
// 已经没有任何数据了,就全部删除
} else if ( isNode ) {
// 如果支持delete,就删除。
// IE使用removeattribute,所以尝试一次。再失败就只能设置为null了。
if ( jQuery.support.deleteExpando ) {
delete elem[ jQuery.expando ];
} else if ( elem.removeAttribute ) {
elem.removeAttribute( jQuery.expando );
} else {
elem[ jQuery.expando ] = null;
}
}
}
复制代码 代码如下:
JQuery.extend({
// 存储数据的地方,关键实现核心
CAChe: { },
// 分配ID用的seed
uUId: 0,
// 为了区别不同的jQuery实例存储的数据,使用前缀+jquery版本号+随机数作为Key
expando: "jquery" + (JQuery.fn.jquery + Math.ranDOM()).Replace(/\D/g, ""),
// 以下元素没有Data:embed和applet(这玩意还活着么),除了FLASH之外的Object。
noData: {
"embed": true,
"object": "clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",
"Applet": true
}
});
对外的接口都调用了两个内部函数:jQuery.data(elem, name, data, pvt)和jQuery.removeData(elem, name, pvt)。而removeData的逻辑与data类似,只是data是加入数据,而removeData使用delete或者设置为null删除数据。
data部分的代码中明确区分了js对象和dom对象的保存,这是为了解决部分浏览器的内存泄漏问题。在低版本ie中,当DOM和JS对象之间出现循环引用时,GC就无法正确处理。参见Understanding and Solving Internet Explorer Leak Patterns。至于COM对象,因为已经限制object元素没有data,就绕过了这个问题。
复制代码 代码如下:
data: function(elem, name, data, pvt) {
// 如果属于noData中定义的元素
if(!jQuery.acceptData(elem)) {
return;
}
varinternalKey = jQuery.expando,
getByName = typeof name === "string",
thisCache,
isnode = elem.nodeType,
// DOM元素需要保存在Cache,js对象直接保存到elem
cache = isNode ? jQuery.cache : elem,
// 如果elem的jQuery.expando已经有值了,就重用
id = isNode ? elem[jQuery.expando] : elem[jQuery.expando] && jQuery.expando;
<PRE class=brush:;gutter:true;><Code>// data未定义,说明当前调用是查询数据,但是对象没有任何数据,直接返回
if((!id || (pvt && id && !cache[id][internalKey])) && getByName && data === undefined) {
return;
}
if(!id) {
if(isNode) {
// 用uuid种子递增分配唯一ID,只有DOM元素需要。因为需要存在全局cache中
elem[jQuery.expando] = id = ++jQuery.uuid;
} else {
id = jQuery.expando;
}
}
// 清空原来的值
if(!cache[id]) {
cache[id] = {};
if(!isNode) {
cache[id].tojson = jQuery.noop;
}
}
// 用extend扩展cache,增加一个属性,用来保存数据
if(typeof name === "object" || typeof name === "function") {
if(pvt) {
cache[id][internalKey] = jQuery.expand(cache[id][internalKey], name);
} else {
cache[id] = jQuery.extend(cache[id], name);
}
}
thisCache = cahce[id];
// 避免Key冲突
if(pvt) {
if(!thisCache[internalKey]) {
thisCahce[internalKey] = {};
}
thisCache = thisCache[internalKey];
}
if(data !== undefined) {
thisCache[jQuery.camelCase(name)] = data;
}
return getByName ? thisCache[jQuery.camelCase(name)] : thisCache;
}
removeData: function( elem, name, pvt ) { // 前面部分与data类似 // ... // 部分浏览器不支持在Element上进行delete操作,在jQuery.support中检查过这个浏览器特性。 // 如果delete失败的话,就先设置成Null。 if ( jQuery.support.deleteExpando || cache != window ) { delete cache[ id ]; } else { cache[ id ] = null; }
<PRE class=brush:;gutter:true;><code>var internalCache = cache[ id ][ internalKey ];
// 如果还有数据,就清空一次再设置,增加性能
if ( internalCache ) {
cache[ id ] = {};
cache[ id ][ internalKey ] = internalCache;
// 已经没有任何数据了,就全部删除
} else if ( isNode ) {
// 如果支持delete,就删除。
// IE使用removeattribute,所以尝试一次。再失败就只能设置为null了。
if ( jQuery.support.deleteExpando ) {
delete elem[ jQuery.expando ];
} else if ( elem.removeAttribute ) {
elem.removeAttribute( jQuery.expando );
} else {
elem[ jQuery.expando ] = null;
}
}
}
相关推荐
- 基于Jquery实现表格动态分页实现代码 - Web前端
- 读jQuery之六 缓存数据功能介绍 - Web前端
- 超轻量级的基于jquery的三级展开列表 - Web前端
- 使用jquery中height()方法获取各种高度大全 - Web前端
- jQuery 操作option的实现代码 - Web前端
- JQuery自适应IFrame高度(支持嵌套 兼容IE,ff,safafi,chrome) - Web前端
- jQuery.autocomplete 支持中文输入(firefox)修正方法 - Web前端
- 基于JQuery的浮动DIV显示提示信息并自动隐藏 - Web前端
- jquery实现控制表格行高亮实例 - Web前端
- jquery 学习之一 对象访问 - Web前端
- 网页前端技术排行
-
- 1【第六章】Foundation之按钮和下拉功能 - Web前端
- 2jQuery编写widget的一些技巧分享 - Web前端
- 3在Mac/PC上远程调试iPhone/iPad上的网页 - Web前端
- 4基于jquery的滚动条滚动固定div(附演示下载) - Web前端
- 5分析Iconfont-阿里巴巴矢量常用图标库 - Web前端
- 6jQuery实例教程:制作网页中可折叠的面板 - Web前端
- 7分享精心挑选的12款优秀jQuery Ajax分页插件和教程 - Web前端
- 8[Web前端]用javascript实现默认图片替代未显示的图片 - Web前端
- 9JS网页制作实例:标签云 - Web前端
- 最近发表
-
- WordPress随机显示特色图片插件:Random Post Thumbnails
- KeePass实现Chrome浏览器自动填充密码方法一
- LNMP一键包nginx 301强制跳转到https教程
- KeePass实现Chrome浏览器自动填充密码方法二
- #建站# 免费的VPS管理软件Xshell8/Xftp8中文版下载
- 使用Xshell 8连接VPS教程_电脑登录vps的方法
- WordPress评论界面添加烟花????效果
- 不同浏览器书签同步方案:坚果云+Floccus_详细使用教程
- iOS端KeePassXC客户端APP:Strongbox Password Safe
- 给WordPress评论中的Gravatar头像图片添加ALT属性


