jQuery數(shù)據(jù)緩存功能的實(shí)現(xiàn)思路及簡(jiǎn)單模擬
對(duì)于jQuery的數(shù)據(jù)緩存,相信大家都不會(huì)陌生,jQuery緩存系統(tǒng)不僅運(yùn)用于DOM元素,動(dòng)畫、事件等都有用到這個(gè)緩存系統(tǒng)。所以在平時(shí)實(shí)際應(yīng)用中, 我們經(jīng)常需要給元素緩存一些數(shù)據(jù),并且這些數(shù)據(jù)往往和DOM元素緊密相關(guān)。由于DOM元素(節(jié)點(diǎn))也是對(duì)象, 所以我們可以直接擴(kuò)展DOM元素的屬性,但是如果給DOM元素添加自定義的屬性和過(guò)多的數(shù)據(jù)可能會(huì)引起內(nèi)存泄漏,所以應(yīng)該要盡量避免這樣做。 因此更好的解決方法是使用一種低耦合的方式讓DOM和緩存數(shù)據(jù)能夠聯(lián)系起來(lái)。
另外:對(duì)于jQuery.data和jQuery.removeData靜態(tài)方法、以及基于這兩個(gè)方法的原型擴(kuò)展方法的介紹和用法就不多說(shuō)了,可以查看官方API文檔。
實(shí)現(xiàn)思路
jQuery提供了一套靈活和強(qiáng)大的緩存方法:
(1)先在jQuery內(nèi)部創(chuàng)建一個(gè)cache對(duì)象{}, 來(lái)保存緩存數(shù)據(jù)。 然后往需要進(jìn)行緩存的DOM節(jié)點(diǎn)上擴(kuò)展一個(gè)值為expando的屬性, 這里是”jQuery” + (new Date).getTime()。 注:expando的值等于”jQuery”+當(dāng)前時(shí)間, 元素本身具有這種屬性的可能性很少,所以可以忽略沖突。
(2)接著把每個(gè)節(jié)點(diǎn)的dom[expando]的值都設(shè)為一個(gè)自增的變量id,保持全局唯一性。 這個(gè)id的值就作為cache的key用來(lái)關(guān)聯(lián)DOM節(jié)點(diǎn)和數(shù)據(jù)。也就是說(shuō)cache[id]就取到了這個(gè)節(jié)點(diǎn)上的所有緩存,即id就好比是打開一個(gè)房間(DOM節(jié)點(diǎn))的鑰匙。 而每個(gè)元素的所有緩存都被放到了一個(gè)map映射里面,這樣可以同時(shí)緩存多個(gè)數(shù)據(jù)。
(3)所以cache對(duì)象結(jié)構(gòu)應(yīng)該像下面這樣:
var cache = {
"uuid1": { // DOM節(jié)點(diǎn)1緩存數(shù)據(jù),"uuid1"相當(dāng)于dom1[expando]
"name1": value1,
"name2": value2
},
"uuid2": { // DOM節(jié)點(diǎn)2緩存數(shù)據(jù),“uuid2"相當(dāng)于dom2[expando]
"name1": value1,
"name2": value2
}
// ......
};
每個(gè)uuid對(duì)應(yīng)一個(gè)elem緩存數(shù)據(jù),每個(gè)緩存對(duì)象是可以由多個(gè)name/value(名值對(duì))對(duì)組成的,而value是可以是任何數(shù)據(jù)類型的。
簡(jiǎn)單模擬實(shí)現(xiàn)
根據(jù)以上思路,就可以簡(jiǎn)單實(shí)現(xiàn)下jQuery.data和jQuery.removeDate的功能了:
(function(window, undefined) {
var cacheData = {}, // 用來(lái)存儲(chǔ)數(shù)據(jù)的對(duì)象
win = window, // 把window緩存給一個(gè)變量
uuid = 0,
// 聲明隨機(jī)數(shù)(8位)
// 注意+new Date()生成的隨機(jī)數(shù)是Number類型,與一個(gè)空字符串連接后(或使用toString方法轉(zhuǎn)型后)變成字符串,才可使用slice方法。
expando = "cacheData" + (+new Date() + "").slice(-8);
// (+new Date()).toString().slice(-8)等價(jià)于expando
// 寫入緩存
var data = function(elem, name, value) {
// 或使用原生方法驗(yàn)證字符串Object.prototype.toString.call(elem) === "[object String]"
// 如果elem為字符串
if (typeof elem === "string") {
// 如果傳入name參數(shù),則為寫入緩存
if (name !== undefined) {
cacheData[elem] = name;
}
// 返回緩存數(shù)據(jù)
return cacheData[elem];
// 如果elem為DOM節(jié)點(diǎn)
} else if (typeof elem === "object") {
var id,
thisCache;
// 如果elem不存在expando屬性,則添加一個(gè)expando屬性(第一次給元素設(shè)置緩存),否則直接獲取已有的expando和id值
if (!elem[expando]) {
id = elem[expando] = ++uuid;
thisCache = cacheData[id] = {};
} else {
id = elem[expando];
thisCache = cacheData[id];
}
// 把一個(gè)隨機(jī)數(shù)作為當(dāng)前緩存對(duì)象的一個(gè)屬性,利用該隨機(jī)數(shù)就能找到該緩存對(duì)象
if (!thisCache[expando]) {
thisCache[expando] = {};
}
if (value !== undefined) {
// 將數(shù)據(jù)存到緩存對(duì)象中
thisCache[expando][name] = value;
}
// 返回DOM元素存儲(chǔ)的數(shù)據(jù)
return thisCache[expando][name];
}
};
// 刪除緩存
var removeData = function(elem, name) {
// 如果elem為字符串,則直接刪除該屬性值
if (typeof elem === "string") {
delete cacheData[elem];
// 如果key為DOM節(jié)點(diǎn)
} else if (typeof elem === "object") {
// 如果elem不存在expando屬性,則終止執(zhí)行,不用刪除緩存
if (!elem[expando]) {
return;
}
// 檢測(cè)對(duì)象是否為空
var isEmptyObject = function(obj) {
var name;
for (name in obj) {
return false;
}
return true;
}
removeAttr = function() {
try {
// IE8即標(biāo)準(zhǔn)瀏覽器可以直接使用delete來(lái)刪除屬性
delete elem[expando];
} catch (e) {
// IE6/IE7使用removeAttribute方法來(lái)刪除屬性
elem.removeAttribute(expando);
}
},
id = elem[expando];
if (name) {
// 只刪除指定的數(shù)據(jù)
delete cacheData[id][expando][name];
// 如果是空對(duì)象,id所對(duì)應(yīng)的數(shù)據(jù)對(duì)象全部刪除
if (isEmptyObject(cacheData[id][expando])) {
delete cacheData[id];
removeAttr();
}
} else {
// 刪除DOM元素存到緩存中的所有數(shù)據(jù)
delete cacheData[id];
removeAttr();
}
}
};
// 把data和removeData掛在window全局對(duì)象下,這樣在外部也能訪問(wèn)到這兩個(gè)函數(shù)
win.expando = expando;
win.data = data;
win.removeData = removeData;
})(window, undefined);
例子:
HTML結(jié)構(gòu):
<div id="demo" style="height: 100px; width: 100px; background: #ccc; color: #fff; margin: 20px; text-align: center; line-height: 100px;">
demo
</div>
js代碼:
window.onload = function() {
// 測(cè)試
var demo = document.getElementById("demo");
// 寫入緩存
data(demo, "myName", "hcy");
console.log(data(demo, "myName")); // hcy
data(demo, "myBlog", "http://www.cnblogs.com/cyStyle");
console.log(data(demo, "myBlog")); // http://www.cnblogs.com/cyStyle
// 刪除DOM元素的某個(gè)緩存值
removeData(demo, "myBlog");
console.log(data(demo, "myBlog")); // undefined
console.log(data(demo, "myName")); // hcy
console.log(demo[expando]); // 1
// 刪除DOM元素
removeData(demo);
console.log(demo[expando]); // undefined
};
firefox下例子結(jié)果截圖:
對(duì)于上述例子實(shí)現(xiàn)jQuery的簡(jiǎn)單緩存系統(tǒng):先給該DOM元素添加一個(gè)隨機(jī)生成的屬性expando,這個(gè)屬性用來(lái)存放訪問(wèn)緩存數(shù)據(jù)的id值,就好比DOM元素都有一把開啟緩存保險(xiǎn)箱的鑰匙,只要有了鑰匙就可以隨時(shí)開啟緩存保險(xiǎn)箱。 將本來(lái)存放到DOM元素中的數(shù)據(jù)都轉(zhuǎn)到了緩存中,而DOM元素本身只要存儲(chǔ)一個(gè)簡(jiǎn)單的屬性就可以了,這樣就可以將由DOM元素引起的內(nèi)存泄漏(具體會(huì)發(fā)生什么狀況不知道,大家都這么說(shuō)~)的風(fēng)險(xiǎn)規(guī)避到最小。
結(jié)語(yǔ)
糊里糊涂地又到了最后,有一些術(shù)語(yǔ)或解釋上可能存在偏差,望各位童鞋指正和給出一些建議;另外,從理論上講, data和removeData方法可以用于任何對(duì)象的緩存, 不過(guò)如果運(yùn)用于本地對(duì)象或window對(duì)象, 會(huì)存在內(nèi)存泄露、循環(huán)引用等問(wèn)題(^_^從網(wǎng)上看到的), 所以一般還是用于DOM節(jié)點(diǎn)比較適合,還可以結(jié)合事件、動(dòng)畫對(duì)DOM節(jié)點(diǎn)進(jìn)行緩存數(shù)據(jù)的操作。ps:cache真的很重要!需要慢慢體會(huì)~
因?yàn)榉窒?,所以?jiǎn)單;因?yàn)榉窒?,所以快?lè)。
- 讀jQuery之六 緩存數(shù)據(jù)功能介紹
- Jquery中Ajax 緩存帶來(lái)的影響的解決方法
- jQuery ajax cache緩存問(wèn)題
- jQuery 數(shù)據(jù)緩存data(name, value)詳解及實(shí)現(xiàn)
- 禁止JQuery中的load方法裝載IE緩存中文件的方法
- jQuery中ajax的使用與緩存問(wèn)題的解決方法
- jquery 緩存問(wèn)題的幾個(gè)解決方法
- IE下jquery ajax無(wú)法獲得最新數(shù)據(jù)的問(wèn)題解決(IE緩存)
- jQuery對(duì)象數(shù)據(jù)緩存Cache原理及jQuery.data方法區(qū)別介紹
- 關(guān)于jQuery對(duì)象數(shù)據(jù)緩存Cache原理以及jQuery.data詳解
- ie下jquery.getJSON的緩存問(wèn)題的處理方法
- jQuery中數(shù)據(jù)緩存$.data的用法及源碼完全解析
相關(guān)文章
jquery滾動(dòng)條插件slimScroll使用方法
這篇文章主要為大家詳細(xì)介紹了jquery滾動(dòng)條插件slimScroll的使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02
jquery $(document).ready() 與window.onload的區(qū)別
Jquery中$(document).ready()的作用類似于傳統(tǒng)JavaScript中的window.onload方法,不過(guò)與window.onload方法還是有區(qū)別的。2009-12-12
jquery實(shí)現(xiàn)的簡(jiǎn)單輪播圖功能【適合新手】
這篇文章主要介紹了jquery實(shí)現(xiàn)的簡(jiǎn)單輪播圖功能,涉及jQuery基于定時(shí)器的事件響應(yīng)與頁(yè)面元素屬性動(dòng)態(tài)操作相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2018-08-08
jquery api參考 visualjquery 中國(guó)線路 速度快
jquery api參考 visualjquery 中國(guó)線路 速度快...2007-11-11
基于jQuery Easyui實(shí)現(xiàn)登陸框界面
本文通過(guò)實(shí)例代碼給大家分享了基于jQuery Easyui實(shí)現(xiàn)登陸框界面,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有參考借鑒價(jià)值,需要的的朋友參考下吧2017-07-07
從零開始學(xué)習(xí)jQuery (五) jquery事件與事件對(duì)象
事件是腳本編程的靈魂. 所以本章內(nèi)容也是jQuery學(xué)習(xí)的重點(diǎn). 本文將對(duì)jQuery中的事件處理以及事件對(duì)象進(jìn)行詳細(xì)的講解.2011-02-02
jquery 刪除節(jié)點(diǎn) 添加節(jié)點(diǎn) 找兄弟節(jié)點(diǎn)的簡(jiǎn)單實(shí)現(xiàn)
下面小編就為大家?guī)?lái)一篇jquery 刪除節(jié)點(diǎn) 添加節(jié)點(diǎn) 找兄弟節(jié)點(diǎn)的簡(jiǎn)單實(shí)現(xiàn)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12
Jquery插件實(shí)現(xiàn)點(diǎn)擊獲取驗(yàn)證碼后60秒內(nèi)禁止重新獲取
這篇文章主要介紹了Jquery插件實(shí)現(xiàn)點(diǎn)擊獲取驗(yàn)證碼后60秒內(nèi)禁止重新獲取,十分常用的功能,這里分享給大家,有需要的小伙伴參考下吧。2015-03-03

