国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

jQuery 數(shù)據(jù)緩存data(name, value)詳解及實現(xiàn)

2019-11-21 00:50:24
字體:
來源:轉載
供稿:網友
作為一名程序員,一提到“緩存”你很容易聯(lián)想到“客戶端(瀏覽器緩存)”和“服務器緩存”。客戶端緩存是存在瀏覽者電腦硬盤上的,即瀏覽器臨時文件夾,而服務器緩存是存在服務器內存中,當然在一些高級應用場合也有專門的緩存服務器,甚至有利用數(shù)據(jù)庫進行緩存的實現(xiàn)。當然這些都不在本文的討論范圍,本文要討論的是最流行的JavaScript框架jQuery的數(shù)據(jù)緩存實現(xiàn)原理,這是jQuery1.2.3版開始加入的新功能。
一、 jQuery數(shù)據(jù)緩存的作用
jQuery數(shù)據(jù)緩存的作用在中文API中是這樣描述的:“用于在一個元素上存取數(shù)據(jù)而避免了循環(huán)引用的風險”。如何理解這句話呢,看看我下面的舉例,不知道合不合適,如果你有更好的例子可以告訴我。
(1) 存在循環(huán)引用風險的例子(注意getDataByName(name)方法中的for in語句):
復制代碼 代碼如下:

<a href="javascript:void(0);" onclick="showInfoByName(this.innerHTML);">Tom</a><br/>
<a href="javascript:void(0);" onclick="showInfoByName(this.innerHTML);">Mike</a>
<script type="text/javascript">
var userInfo = [
{
"name": "Tom",
"age": 21,
"phone": "020-12345678"
},
{
"name": "Mike",
"age": 23,
"phone": "020-87654321"
}];
function getDataByName(name)
{
for (var i in userInfo)
{
if (userInfo[i].name == name)
{
return userInfo[i];
break;
}
}
}
function showInfoByName(name)
{
var info = getDataByName(name);
alert('name:' + info.name + '/n' + 'age:' + info.age + '/n' + 'phone:' + info.phone);
}
</script>

(2) 優(yōu)化循環(huán)引用風險的例子(本例子其實與jQuery緩存實現(xiàn)原理差不多了,本例子重點在于改寫了userInfo這個JSON結構,使name與對象key直接對應):
復制代碼 代碼如下:

<a href="javascript:void(0);" onclick="showInfoByName(this.innerHTML);">Tom</a><br/>
<a href="javascript:void(0);" onclick="showInfoByName(this.innerHTML);">Mike</a>
<script type="text/javascript">
var userInfo =
{
"Tom":
{
"name": "Tom",
"age": 21,
"phone": "020-12345678"
},
"Mike":
{
"name": "Mike",
"age": 23,
"phone": "020-87654321"
}
};
function showInfoByName(name)
{
var info = userInfo[name];
alert('name:' + info.name + '/n' + 'age:' + info.age + '/n' + 'phone:' + info.phone);
}
</script>

二、簡單實現(xiàn)jQuery設置數(shù)據(jù)緩存方法
jQuery數(shù)據(jù)緩存的實現(xiàn)其實是很簡單的,下面我來實現(xiàn)jQuery設置數(shù)據(jù)緩存方法,我讓代碼盡量的簡單,這有助于你更容易了解data的實現(xiàn)原理。函數(shù)與測試代碼如下:
復制代碼 代碼如下:

<div id="div1">div1</div><br/>
<div id="div2">div2</div>
<script type="text/javascript">
//cache對象結構像這樣{"uuid1":{"name1":value1,"name2":value2},"uuid2":{"name1":value1,"name2":value2}},每個uuid對應一個elem緩存數(shù)據(jù),每個緩存對象是可以由多個name/value對組成的,而value是可以是任何數(shù)據(jù)類型的,比如可以像這樣在elem下存一個JSON片段:$(elem).data('JSON':{"name":"Tom","age":23})
var cache = {};
//expando作為elem一個新加屬性,為了防止與用戶自己定義的產生沖突,這里采用可變后綴
var expando = 'jQuery' + new Date().getTime();
var uuid = 0;
function data(elem, name, data)
{
//至少保證要有elem和name兩個參數(shù)才能進行取緩存或設置緩存操作
if (elem && name)
{
//嘗試取elem標簽expando屬性
var id = elem[expando];
if (data)
{
//設置緩存數(shù)據(jù)
if (!id)
id = elem[expando] = ++uuid;
//如果cache中id鍵對象不存在(即這個elem沒有設置過數(shù)據(jù)緩存),先創(chuàng)建一個空對象
if (!cache[id])
cache[id] = {};
cache[id][name] = data;
}
else
{
//獲取緩存數(shù)據(jù)
if (!id)
return 'Not set cache!';
else
return cache[id][name];
}
}
}
var div = document.getElementById('div1');
data(div, "tagName", "div");
data(div, "ID", "div1");
alert(data(div, "tagName")); //div
alert(data(div, "ID")); //div1
var div2 = document.getElementById('div2');
alert(data(div2, "tagName")); //Not set cache!
</script>

三、使用jQuery數(shù)據(jù)緩存注意事項
(1)因為jQuery緩存對象是全局的,在AJAX應用中,由于頁面刷新很少,這個對象將一直存在,隨著你對data的不斷操作,很有可能因為使用不當,使得這個對象不斷變大,最終影響程序性能。所以我們要及時清理這個對象,jQuery也提供了相應方法:removeData(name),name就是你當初設置data值時使用的name參數(shù)。
另外,根據(jù)我對jQuery代碼的了解,發(fā)現(xiàn)下面幾種情況不需要手動清除數(shù)據(jù)緩存:
<1> 對elem執(zhí)行remove()操作,jQuery會清除對象可能存在的緩存。jQuery相關源代碼參考:
復制代碼 代碼如下:

remove:function(selector)
{
if (!selector || jQuery.filter(selector, [this]).length)
{
// Prevent memory leaks
jQuery("*", this).add([this]).each(function()
{
jQuery.event.remove(this);
jQuery.removeData(this);
});
if (this.parentNode)
this.parentNode.removeChild(this);
}
}

<2> 對elem執(zhí)行empty()操作,如果當前elem子元素存在數(shù)據(jù)緩存,jQuery也會清除子對象可能存在的數(shù)據(jù)緩存,因為jQuery的empty()實現(xiàn)其實是循環(huán)調用remove()刪除子元素。jQuery相關源代碼參考:
復制代碼 代碼如下:

empty:function()
{
// Remove element nodes and prevent memory leaks
jQuery(this).children().remove();
// Remove any remaining nodes
while (this.firstChild)
this.removeChild(this.firstChild);
}

2、jQuery復制節(jié)點clone()方法不會復制data緩存,準確說jQuery不會在全局緩存對象中分配一個新節(jié)點存放新復制elem緩存。jQuery在clone()中把可能存在的緩存指向屬性(elem的expando屬性)替換成空。如果直接把這個屬性復制,就會導致原先和新復制的elem都指向一個數(shù)據(jù)緩存,中間的互操作都將會影響到兩個elem的緩存變量。以下jQuery代碼就是把expando屬性刪除(jQuery1.3.2,較早版本不是這樣處理,顯然新版本的這個方法性能更好)。
jQuery.clean([html.replace(/ jQuery/d+="(?:/d+|null)"/g, "").replace(/^/s*/, "")])[0];
把數(shù)據(jù)緩存一起復制有時候也是很有用的,比如在拖動操作中,我們點擊源目標elem節(jié)點就會復制出一個半透明的elem副本開始拖動,并把data緩存復制到拖動層中,等到拖動結束,我們就可能取到當前拖動的elem相關信息。現(xiàn)在jQuery方法沒有給我們提供這樣的處理,怎么辦法。第一個辦法是改寫jQuery代碼,這個方法顯然很傻,很不科學。正確做法是復制源目標的data,把這些data都重新設置到復制出來的elem中,這樣在執(zhí)行data(name, value)方法時,jQuery會在全局緩存對象中為我們開辟新空間。實現(xiàn)代碼如下:
復制代碼 代碼如下:

if (typeof($.data(currentElement)) == 'number')
{
var elemData = $.cache[$.data(currentElement)];
for (var k in elemData)
{
dragingDiv.data(k, elemData[k]);
}
}

在上面代碼中,$.data(elem,name,data)包含三個參數(shù),如果只有一個elem參數(shù),這個方法返回它的緩存key(即uuid),利用這個key就可以得到整個緩存對象,然后把對象的數(shù)據(jù)都復制到新的對象。
發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 炉霍县| 织金县| 平泉县| 磐安县| 湖南省| 利津县| 内江市| 合肥市| 吉木萨尔县| 巴里| 会理县| 武宁县| 东丰县| 新昌县| 昭苏县| 长阳| 镇安县| 邵东县| 赞皇县| 奉节县| 翁牛特旗| 和田县| 瑞昌市| 临邑县| 满洲里市| 郯城县| 阳山县| 涟水县| 汉川市| 寻甸| 手游| 无锡市| 丰县| 正镶白旗| 荆州市| 高尔夫| 寿光市| 吐鲁番市| 辉南县| 河池市| 新闻|