理解和運用JavaScript的閉包機制
偉大的愛因斯坦同志說過:“如果你無法向一個 6 歲小孩解釋清楚某問題,那說明你自己都沒整明白”。然而,當我向一個 27 歲的朋友解釋什么是閉包時,卻徹底失敗了。
這原本是國外某哥們兒在 Stack Overflow 上對 JavaScript 閉包所提出的問題。不過既然此問題是在 Stack Overflow 提出的,當然也會有很多高手出來解答,其中有些回答確實是經典,如下面這個:
如果在一個外部函數中再定義一個內部函數,即函數嵌套函數,那么內部函數也可以訪問外部函數中的變量:
function foo(x) {
var tmp = 3;
function bar(y) {
alert(x + y + (++tmp));
}
bar(10);
}
foo(2); // alert 16
foo(2); // alert 16
foo(2); // alert 16
此段代碼可以正確執(zhí)行,并返回結果:16,因為 bar 能訪問外部函數的變量 tmp, 同時也能訪問外部函數 foo 的參數 x。但以上示例不是閉包!
要實現閉包的話,需要將內部函數作為外部函數的返回值返回,內部函數在返回前,會將所有已訪問過的外部函數中的變量在內存中鎖定,也就是說,這些變量將常駐 bar 的內存中,不會被垃圾回收器回收,如下:
function foo(x) {
var tmp = 3;
return function (y) {
alert(x + y + (++tmp));
}
}
var bar = foo(2); // bar 現在是個閉包了
bar(10); // alert 16
bar(10); // alert 17
bar(10); // alert 18
上述代碼中,第一次執(zhí)行 bar 時,仍會返回結果:16,因為 bar 仍然可以訪問 x 及 tmp,盡管它已經不直接存在于 foo 的作用域內。那么既然 tmp 被鎖定在 bar 的閉包里,那么每次執(zhí)行 bar 的時候,tmp 都會自增一次,所以第二次和第三次執(zhí)行 bar 時,分別返回 17 和 18。
此示例中,x 僅僅是個純粹的數值,當 foo 被調用時,數值 x 就會作為參數被拷貝至 foo 內。
但是 JavaScript 處理對象的時候,使用的總是引用,如果用一個對象作為參數來調用 foo,那么 foo 中傳入的實際上是原始對象的引用,所以這個原始對象也相當于被閉包了,如下:
function foo(x) {
var tmp = 3;
return function (y) {
alert(x + y + tmp++);
x.memb = x.memb ? x.memb + 1 : 1;
alert(x.memb);
}
}
var age = new Number(2);
var bar = foo(age); // bar 現在是個閉包了
bar(10); // alert 15 1
bar(10); // alert 16 2
bar(10); // alert 17 3
和期望的一樣,每次執(zhí)行 bar(10) 時,不但 tmp 自增了,x.memb 也自增了,因為函數體內的 x 和函數體外的 age 引用的是同一個對象。
via http://stackoverflow.com/questions/111102/how-do-javascript-closures-work
補充:通過以上示例,應該能比較清楚的理解閉包了。如果覺得自己理解了,可以試著猜猜下面這段代碼的執(zhí)行結果:
function foo(x) {
var tmp = 3;
return function (y) {
alert(x + y + tmp++);
x.memb = x.memb ? x.memb + 1 : 1;
alert(x.memb);
}
}
var age = new Number(2);
var bar1 = foo(age); // bar1 現在是個閉包了
bar1(10); // alert 15 1
bar1(10); // alert 16 2
bar1(10); // alert 17 3
var bar2 = foo(age); // bar2 現在也是個閉包了
bar2(10); // alert ? ?
bar2(10); // alert ? ?
bar2(10); // alert ? ?
bar1(10); // alert ? ?
bar1(10); // alert ? ?
bar1(10); // alert ? ?
實際使用的時候,閉包可以創(chuàng)建出非常優(yōu)雅的設計,允許對funarg上定義的多種計算方式進行定制。如下就是數組排序的例子,它接受一個排序條件函數作為參數:
[1, 2, 3].sort(function (a, b) {
... // 排序條件
});
同樣的例子還有,數組的map方法是根據函數中定義的條件將原數組映射到一個新的數組中:
[1, 2, 3].map(function (element) {
return element * 2;
}); // [2, 4, 6]
使用函數式參數,可以很方便的實現一個搜索方法,并且可以支持無限制的搜索條件:
someCollection.find(function (element) {
return element.someProperty == 'searchCondition';
});
還有應用函數,比如常見的forEach方法,將函數應用到每個數組元素:
[1, 2, 3].forEach(function (element) {
if (element % 2 != 0) {
alert(element);
}
}); // 1, 3
順便提下,函數對象的 apply 和 call方法,在函數式編程中也可以用作應用函數。 這里,我們將它們看作是應用函數 —— 應用到參數中的函數(在apply中是參數列表,在call中是獨立的參數):
(function () {
alert([].join.call(arguments, ';')); // 1;2;3
}).apply(this, [1, 2, 3]);
閉包還有另外一個非常重要的應用 —— 延遲調用:
var a = 10;
setTimeout(function () {
alert(a); // 10, after one second
}, 1000);
還有回調函數:
//...
var x = 10;
// only for example
xmlHttpRequestObject.onreadystatechange = function () {
// 當數據就緒的時候,才會調用;
// 這里,不論是在哪個上下文中創(chuàng)建
// 此時變量“x”的值已經存在了
alert(x); // 10
};
//...
還可以創(chuàng)建封裝的作用域來隱藏輔助對象:
var foo = {};
// 初始化
(function (object) {
var x = 10;
object.getX = function _getX() {
return x;
};
})(foo);
alert(foo.getX()); // 獲得閉包 "x" – 10
相關文章
Javascript基礎教程之數據類型 (字符串 String)
javascript一共有9種數據類型,分別是字符串 String、數值型 Number、布爾型 Boolean、未定義 Undefine、空值 Null、對象 Object、引用Refernce、列表型 List、完成型 Completion,我們今天首先來看看(字符串 String)2015-01-01
深入理解JavaScript系列(21):S.O.L.I.D五大原則之接口隔離原則ISP詳解
這篇文章主要介紹了深入理解JavaScript系列(21):S.O.L.I.D五大原則之接口隔離原則ISP詳解,本文講解了JavaScript接口、ISP與JavaScript、墮落的實現、靜態(tài)耦合、語義耦合、可擴展性等內容,需要的朋友可以參考下2015-03-03
JavaScript高級程序設計(第3版)學習筆記10 再訪js對象
在ECMAScript中,兩個核心主題就是對象與函數,而這兩個主題也有些互相纏繞的,在前面幾個博文中大略的過了一遍函數相關的基礎知識,這篇文章再回到對象主題上來2012-10-10

