理解javascript函數(shù)式編程中的閉包(closure)
來源:易賢網(wǎng) 閱讀:859 次 日期:2016-07-21 16:09:33
溫馨提示:易賢網(wǎng)小編為您整理了“理解javascript函數(shù)式編程中的閉包(closure)”,方便廣大網(wǎng)友查閱!

這篇文章主要幫助大家理解javascript函數(shù)式編程中的閉包(closure)概念,通俗地講, JavaScript 中每個的函數(shù)都是一個閉包,感興趣的小伙伴們可以參考一下

閉包(closure)是函數(shù)式編程中的概念,出現(xiàn)于 20 世紀(jì) 60 年代,最早實現(xiàn)閉包的語言是 Scheme,它是 LISP 的一種方言。之后閉包特性被其他語言廣泛吸納。

閉包的嚴(yán)格定義是“由函數(shù)(環(huán)境)及其封閉的自由變量組成的集合體?!边@個定義對于大家來說有些晦澀難懂,所以讓我們先通過例子和不那么嚴(yán)格的解釋來說明什么是閉包,然后再舉例說明一些閉包的經(jīng)典用途。

什么是閉包

通俗地講, JavaScript 中每個的函數(shù)都是一個閉包,但通常意義上嵌套的函數(shù)更能夠體

現(xiàn)出閉包的特性,請看下面這個例子:

var generateClosure = function() {

var count = 0;

var get = function() {

count ++;

return count;

};

return get;

};

var counter = generateClosure();

console.log(counter()); // 輸出 1

console.log(counter()); // 輸出 2

console.log(counter()); // 輸出 3

這段代碼中, generateClosure() 函數(shù)中有一個局部變量count, 初值為 0。還有一個叫做 get 的函數(shù), get 將其父作用域,也就是 generateClosure() 函數(shù)中的 count 變量增加 1,并返回 count 的值。 generateClosure() 的返回值是 get 函數(shù)。在外部我們通過 counter 變量調(diào)用了 generateClosure() 函數(shù)并獲取了它的返回值,也就是 get 函數(shù),接下來反復(fù)調(diào)用幾次 counter(),我們發(fā)現(xiàn)每次返回的值都遞增了 1。

讓我們看看上面的例子有什么特點,按照通常命令式編程思維的理解, count 是generateClosure 函數(shù)內(nèi)部的變量,它的生命周期就是 generateClosure 被調(diào)用的時期,當(dāng) generateClosure 從調(diào)用棧中返回時, count 變量申請的空間也就被釋放。問題是,在 generateClosure() 調(diào)用結(jié)束后, counter() 卻引用了“已經(jīng)釋放了的” count變量,而且非但沒有出錯,反而每次調(diào)用 counter() 時還修改并返回了 count。這是怎么回事呢?

這正是所謂閉包的特性。當(dāng)一個函數(shù)返回它內(nèi)部定義的一個函數(shù)時,就產(chǎn)生了一個閉包,閉 包 不 但 包 括 被 返 回 的 函 數(shù) , 還包括這個函數(shù)的定義環(huán)境。上面例子中,當(dāng)函數(shù)generateClosure() 的內(nèi)部函數(shù) get 被一個外部變量 counter 引用時, counter 和generateClosure() 的局部變量就是一個閉包。如果還不夠清晰,下面這個例子可以幫助

你理解:

var generateClosure = function() {

var count = 0;

var get = function() {

count ++;

return count;

};

return get;

};

var counter1 = generateClosure();

var counter2 = generateClosure();

console.log(counter1()); // 輸出 1

console.log(counter2()); // 輸出 1

console.log(counter1()); // 輸出 2

console.log(counter1()); // 輸出 3

console.log(counter2()); // 輸出 2

上面這個例子解釋了閉包是如何產(chǎn)生的:counter1 和 counter2 分別調(diào)用了 generateClosure() 函數(shù),生成了兩個閉包的實例,它們內(nèi)部引用的 count 變量分別屬于各自的運行環(huán)境。我們可以理解為,在generateClosure() 返回 get 函數(shù)時,私下將 get 可能引用到的 generateClosure() 函數(shù)的內(nèi)部變量(也就是 count 變量)也返回了,并在內(nèi)存中生成了一個副本,之后 generateClosure() 返回的函數(shù)的兩個實例 counter1和 counter2 就是相互獨立的了。

閉包的用途

1、嵌套的回調(diào)函數(shù)

閉包有兩個主要用途,一是實現(xiàn)嵌套的回調(diào)函數(shù),二是隱藏對象的細(xì)節(jié)。讓我們先看下面這段代碼示例,了解嵌套的回調(diào)函數(shù)。如下代碼是在 Node.js 中使用 MongoDB 實現(xiàn)一個簡單的增加用戶的功能:

exports.add_user = function(user_info, callback) {

var uid = parseInt(user_info['uid']);

mongodb.open(function(err, db) {

if (err) {callback(err); return;}

db.collection('users', function(err, collection) {

if (err) {callback(err); return;}

collection.ensureIndex("uid", function(err) {

if (err) {callback(err); return;}

collection.ensureIndex("username", function(err) {

if (err) {callback(err); return;}

collection.findOne({uid: uid}, function(err) {

if (err) {callback(err); return;}

if (doc) {

callback('occupied');

} else {

var user = {

uid: uid,

user: user_info,

};

collection.insert(user, function(err) {

callback(err);

});

}

});

});

});

});

});

};

如果你對 Node.js 或 MongoDB 不熟悉,沒關(guān)系,不需要去理解細(xì)節(jié),只要看清楚大概的邏輯即可。這段代碼中用到了閉包的層層嵌套,每一層的嵌套都是一個回調(diào)函數(shù)?;卣{(diào)函數(shù)不會立即執(zhí)行,而是等待相應(yīng)請求處理完后由請求的函數(shù)回調(diào)。我們可以看到,在嵌套的每一層中都有對 callback 的引用,而且最里層還用到了外層定義的 uid 變量。由于閉包機制的存在,即使外層函數(shù)已經(jīng)執(zhí)行完畢,其作用域內(nèi)申請的變量也不會釋放,因為里層的函數(shù)還有可能引用到這些變量,這樣就完美地實現(xiàn)了嵌套的異步回調(diào)。

2、實現(xiàn)私有成員

我們知道, JavaScript 的對象沒有私有屬性,也就是說對象的每一個屬性都是曝露給外部的。這樣可能會有安全隱患,譬如對象的使用者直接修改了某個屬性,導(dǎo)致對象內(nèi)部數(shù)據(jù)的一致性受到破壞等。 JavaScript通過約定在所有私有屬性前加上下劃線(例如_myPrivateProp),表示這個屬性是私有的,外部對象不應(yīng)該直接讀寫它。但這只是個非正式的約定,假設(shè)對象的使用者不這么做,有沒有更嚴(yán)格的機制呢?答案是有的,通過閉包可以實現(xiàn)。讓我們再看看前面那個例子:

var generateClosure = function() {

var count = 0;

var get = function() {

count ++;

return count;

};

return get;

};

var counter = generateClosure();

console.log(counter()); // 輸出 1

console.log(counter()); // 輸出 2

console.log(counter()); // 輸出 3

我們可以看到,只有調(diào)用 counter() 才能訪問到閉包內(nèi)的 count 變量,并按照規(guī)則對其增加1,除此之外決無可能用其他方式找到 count 變量。受到這個簡單例子的啟發(fā),我們可以把一個對象用閉包封裝起來,只返回一個“訪問器”的對象,即可實現(xiàn)對細(xì)節(jié)隱藏。

以上就是本文的全部內(nèi)容,希望能夠幫助大家更好的學(xué)習(xí)理解javascript閉包。

更多信息請查看網(wǎng)絡(luò)編程
易賢網(wǎng)手機網(wǎng)站地址:理解javascript函數(shù)式編程中的閉包(closure)
由于各方面情況的不斷調(diào)整與變化,易賢網(wǎng)提供的所有考試信息和咨詢回復(fù)僅供參考,敬請考生以權(quán)威部門公布的正式信息和咨詢?yōu)闇?zhǔn)!

2025國考·省考課程試聽報名

  • 報班類型
  • 姓名
  • 手機號
  • 驗證碼
關(guān)于我們 | 聯(lián)系我們 | 人才招聘 | 網(wǎng)站聲明 | 網(wǎng)站幫助 | 非正式的簡要咨詢 | 簡要咨詢須知 | 新媒體/短視頻平臺 | 手機站點 | 投訴建議
工業(yè)和信息化部備案號:滇ICP備2023014141號-1 云南省教育廳備案號:云教ICP備0901021 滇公網(wǎng)安備53010202001879號 人力資源服務(wù)許可證:(云)人服證字(2023)第0102001523號
云南網(wǎng)警備案專用圖標(biāo)
聯(lián)系電話:0871-65099533/13759567129 獲取招聘考試信息及咨詢關(guān)注公眾號:hfpxwx
咨詢QQ:1093837350(9:00—18:00)版權(quán)所有:易賢網(wǎng)
云南網(wǎng)警報警專用圖標(biāo)