在javascript中,閉包是一個(gè)常見(jiàn)且重要的概念,它可以讓我們創(chuàng)建一種特殊的函數(shù),它可以訪問(wèn)其所在的函數(shù)作用域中的變量和函數(shù)。簡(jiǎn)單來(lái)說(shuō),閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)。閉包的作用是令函數(shù)在運(yùn)行時(shí)可以訪問(wèn)其定義時(shí)的上下文環(huán)境。
舉例說(shuō)明,假設(shè)我們有兩個(gè)函數(shù),一個(gè)是父函數(shù),一個(gè)是子函數(shù),它們都定義了一些變量:
function parent() { var name = "parentName"; function child() { console.log(name); } child(); } parent();
當(dāng)我們執(zhí)行parent函數(shù)時(shí),會(huì)先定義name變量并聲明child函數(shù),然后執(zhí)行child函數(shù)。在child函數(shù)中,它會(huì)打印出parent函數(shù)中的name變量。這只是一個(gè)簡(jiǎn)單的閉包例子,其中child函數(shù)就可以訪問(wèn)到parent函數(shù)的作用域。
我們可以使用閉包來(lái)創(chuàng)建一些不會(huì)被外界訪問(wèn)的變量,例如我們可以通過(guò)立即執(zhí)行函數(shù)來(lái)創(chuàng)建一個(gè)計(jì)數(shù)器:
var counter = (function() { var count = 0; return function() { return ++count; } })(); console.log(counter()); // 1 console.log(counter()); // 2 console.log(counter()); // 3
這里我們使用了立即執(zhí)行函數(shù),它返回了一個(gè)函數(shù)并將它賦值給counter變量。在內(nèi)部函數(shù)中,它可以訪問(wèn)外部函數(shù)中定義的count變量,因?yàn)閮?nèi)部函數(shù)形成了一個(gè)閉包,它可以訪問(wèn)外部函數(shù)的作用域,所以在執(zhí)行counter函數(shù)時(shí),它會(huì)返回一個(gè)遞增的計(jì)數(shù)器。
閉包在處理回調(diào)函數(shù)時(shí)也非常常見(jiàn),例如我們可以使用jQuery中的each函數(shù),通過(guò)在回調(diào)函數(shù)中存儲(chǔ)變量來(lái)記錄循環(huán)次數(shù):
var arr = [ "a", "b", "c" ]; var count = 0; $.each(arr, function(index, value) { count++; }); console.log(count); // 3
在上面的例子中,我們使用回調(diào)函數(shù)來(lái)循環(huán)一個(gè)數(shù)組,每當(dāng)回調(diào)函數(shù)被執(zhí)行時(shí),它會(huì)訪問(wèn)外部函數(shù)中定義的count變量并將其遞增。最終輸出count變量,它的值就是數(shù)組的長(zhǎng)度。
總之,閉包是javascript中一個(gè)強(qiáng)大的概念,我們可以使用它來(lái)創(chuàng)建具有私有變量的函數(shù)、進(jìn)行回調(diào)函數(shù)的編程,以及創(chuàng)建一些高級(jí)的設(shè)計(jì)模式,因此我們需要對(duì)閉包有一個(gè)深入的理解。