Javascript閉包用法實(shí)例分析
本文實(shí)例分析了Javascript閉包的概念及用法。分享給大家供大家參考。具體如下:
提到閉包,想必大家都早有耳聞,下面說(shuō)下我的簡(jiǎn)單理解。
說(shuō)實(shí)話(huà)平時(shí)工作中實(shí)際手動(dòng)寫(xiě)閉包的場(chǎng)景并不多,但是項(xiàng)目中用到的第三方框架和組件或多或少用到了閉包。
所以,了解閉包是非常必要的。呵呵...
一、什么是閉包
簡(jiǎn)而言之,就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)。
由于JS變量作用域的特性,外部不能訪問(wèn)內(nèi)部變量,內(nèi)部可以外部變量。
二、使用場(chǎng)景
1. 實(shí)現(xiàn)私有成員。
2. 保護(hù)命名空間,避免污染全局變量。
3. 緩存變量。
先看一個(gè)封裝的例子:
// 變量作用域?yàn)楹瘮?shù)內(nèi)部,外部無(wú)法訪問(wèn)
var name = "default";
return {
getName: function () {
return name;
},
setName: function (newName) {
name = newName;
}
}
}();
console.log(person.name); // 直接訪問(wèn),結(jié)果為:undefined
console.log(person.getName()); // 結(jié)果為:default
console.log(person.setName("langjt"));
console.log(person.getName()); // 結(jié)果為:langjt
再看循環(huán)中常用閉包解決引用外部變量問(wèn)題:
for (var i=0, len=aLi.length; i<len; i++) {
aLi[i].onclick = function() {
alert(i); // 無(wú)論點(diǎn)擊哪個(gè)<li>元素,彈出的值都為len,表明這里的i和在for之后打印i的值是一樣的。
};
}
使用閉包后:
for (var i=0, len=aLi.length; i<len; i++) {
aLi[i].onclick = (function(i) {
return function() {
alert(i); // 此時(shí)點(diǎn)擊<li>元素,就會(huì)彈出對(duì)應(yīng)的下標(biāo)了。
}
})(i);
}
三、注意事項(xiàng)
1. 內(nèi)存泄漏
由于閉包會(huì)使得函數(shù)中的變量都被保存在內(nèi)存中,內(nèi)存消耗很大,所以不能濫用閉包,否則會(huì)造成網(wǎng)頁(yè)的性能問(wèn)題。
比如:
var oDiv = document.getElementById(‘J_DIV');
var id = oDiv.id;
oDiv.onclick = function() {
// alert(oDiv.id); 這里存在循環(huán)引用,IE低版本頁(yè)面關(guān)閉后oDiv仍在內(nèi)存中。所以盡可能緩存基本類(lèi)型而不是對(duì)象。
alert(id);
};
oDiv = null;
}
2. 變量命名
如果內(nèi)部函數(shù)的變量和外部函數(shù)的變量名相同時(shí),那么內(nèi)部函數(shù)再也無(wú)法指向外部函數(shù)那個(gè)同名的變量。
比如:
return function(num) {
console.log(num);
}
}
var f = new foo(9);
f(); // undefined
其實(shí)上面的用法,專(zhuān)業(yè)術(shù)語(yǔ)叫函數(shù)柯里化(Currying),就是把接受多個(gè)參數(shù)的函數(shù)變換成接受一個(gè)單一參數(shù)(最初函數(shù)的第一個(gè)參數(shù))的函數(shù),并且返回接受余下的參數(shù)而且返回結(jié)果的新函數(shù)的技術(shù)。本質(zhì)上也利用了閉包可以緩存的特性,比如:
return function(y) {
return num+y;
};
};
var inc = adder(1);
var dec = adder(-1);
//inc, dec現(xiàn)在是兩個(gè)新的函數(shù),作用是將傳入的參數(shù)值 (+/‐)1
alert(inc(99));//100
alert(dec(101));//100
alert(adder(100)(2));//102
alert(adder(2)(100));//102
再比如阿里玉伯的seaJS源碼中:
* util-lang.js - The minimal language enhancement
*/
function isType(type) {
return function(obj) {
return {}.toString.call(obj) == "[object " + type + "]"
}
}
var isObject = isType("Object");
var isString = isType("String");
希望本文所述對(duì)大家的javascript程序設(shè)計(jì)有所幫助。
相關(guān)文章
微信小程序之?dāng)?shù)據(jù)緩存的實(shí)例詳解
這篇文章主要介紹了微信小程序之?dāng)?shù)據(jù)緩存的實(shí)例詳解的相關(guān)資料,希望通過(guò)本文能幫助到大家,讓大家掌握這部分內(nèi)容,需要的朋友可以參考下2017-09-09判斷js的Array和Object的實(shí)現(xiàn)方法
下面小編就為大家?guī)?lái)一篇判斷js的Array和Object的實(shí)現(xiàn)方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-08-08JavaScript設(shè)計(jì)模式之單例模式詳解
這篇文章主要為大家詳細(xì)介紹了JavaScript設(shè)計(jì)模式之例模式的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06JS把字符串格式的時(shí)間轉(zhuǎn)換成幾秒前、幾分鐘前、幾小時(shí)前、幾天前等格式
最近在做項(xiàng)目的時(shí)候,需要把后臺(tái)返回的時(shí)間轉(zhuǎn)換成幾秒前、幾分鐘前、幾小時(shí)前、幾天前等的格式,接下來(lái)通過(guò)本文給大家分享JS把字符串格式的時(shí)間轉(zhuǎn)換成幾秒前、幾分鐘前、幾小時(shí)前、幾天前等格式 ,需要的朋友可以參考下2019-07-07javascript 動(dòng)態(tài)參數(shù)判空操作
在做交友中心的頁(yè)面的時(shí)候,有一個(gè)javascript函數(shù),它的第二個(gè)參數(shù)是動(dòng)態(tài)的。2008-12-12JavaScript的arguments對(duì)象應(yīng)用示例
使用特殊對(duì)象 arguments,開(kāi)發(fā)者無(wú)需明確指出參數(shù)名,就能訪問(wèn)它們,下面為大家介紹下其具體的應(yīng)用2014-09-09JS判斷來(lái)路是否是百度等搜索索引進(jìn)行彈窗或自動(dòng)跳轉(zhuǎn)的實(shí)現(xiàn)代碼
這篇文章主要介紹了JS判斷來(lái)路是否是百度等搜索索引進(jìn)行彈窗或自動(dòng)跳轉(zhuǎn)的實(shí)現(xiàn)代碼的相關(guān)資料,需要的朋友可以參考下2016-10-10