譯者按: 從最簡(jiǎn)單的計(jì)數(shù)器開(kāi)始,按照需求對(duì)代碼一步步優(yōu)化,我們可以領(lǐng)會(huì)閉包的神奇之處。
成都創(chuàng)新互聯(lián)公司-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比襄垣網(wǎng)站開(kāi)發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫(kù),直接使用。一站式襄垣網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋襄垣地區(qū)。費(fèi)用合理售后完善,十載實(shí)體公司更值得信賴。原文: Closures are not magic
對(duì)于JavaScript新手來(lái)說(shuō),閉包(Closures)是一個(gè)很神奇的東西。這篇博客將通過(guò)一個(gè)非常淺顯的代碼示例來(lái)解釋閉包。
我們的目標(biāo)是實(shí)現(xiàn)一個(gè)計(jì)數(shù)器,它的效果如下:
increment(); // Number of events: 1
increment(); // Number of events: 2
increment(); // Number of events: 3
可知,每次執(zhí)行increment()都會(huì)輸出"Number of events: N",且N每次都會(huì)加1。
這個(gè)計(jì)數(shù)器最直觀的實(shí)現(xiàn)方式如下:
var counter = 0;
function increment()
{
counter = counter + 1;
console.log("Number of events: " + counter);
}
以上的代碼非常簡(jiǎn)單。但是,當(dāng)我們需要第二個(gè)計(jì)數(shù)器時(shí),就會(huì)遇到問(wèn)題了。當(dāng)然,我們可以實(shí)現(xiàn)兩個(gè)重復(fù)的計(jì)數(shù)器:
var counter1 = 0;
function incrementCounter1()
{
counter1 = counter1 + 1;
console.log("Number of events: " + counter1);
}
var counter2 = 0;
function incrementCounter2()
{
counter2 = counter2 + 1;
console.log("Number of events: " + counter2);
}
incrementCounter1(); // Number of events: 1
incrementCounter2(); // Number of events: 1
incrementCounter1(); // Number of events: 2
顯然,以上的代碼非常冗余,有待優(yōu)化。當(dāng)我們需要更多計(jì)數(shù)器時(shí),使用這種方法將不太現(xiàn)實(shí)。這時(shí),就需要神奇的閉包了。
需要多個(gè)計(jì)數(shù)器,同時(shí)希望去除冗余代碼的話,就可以使用閉包了:
function createCounter()
{
var counter = 0;
function increment()
{
counter = counter + 1;
console.log("Number of events: " + counter);
}
return increment;
}
var counter1 = createCounter();
var counter2 = createCounter();
counter1(); // Number of events: 1
counter1(); // Number of events: 2
counter2(); // Number of events: 1
counter1(); // Number of events: 3
在代碼中,我們創(chuàng)建了兩個(gè)獨(dú)立的計(jì)數(shù)器counter1與counter2,分別進(jìn)行計(jì)數(shù),互不干撓。代碼看著有點(diǎn)奇怪,我們不妨拆分起來(lái)分析。
首先,我們來(lái)看看createCounter:
看起來(lái),createCounter()函數(shù)與我們最初定義的計(jì)數(shù)器非常相似。唯一的不同點(diǎn)在于:createCounter()將計(jì)數(shù)器封裝在一個(gè)函數(shù)內(nèi),于是我們將它稱作閉包。
難以理解的一點(diǎn)在于,當(dāng)我們使用createCounter()函數(shù)創(chuàng)建計(jì)數(shù)器時(shí),實(shí)際上創(chuàng)建了一個(gè)新的函數(shù):
// fancyNewCounter是一個(gè)新創(chuàng)建的函數(shù)
var fancyNewCounter = createCounter();
閉包的神奇之處在于。每次使用createCounter()函數(shù)創(chuàng)建計(jì)數(shù)器increment時(shí),都會(huì)創(chuàng)建一個(gè)對(duì)應(yīng)的counter變量。并且,返回的increment函數(shù)會(huì)始終記住counter變量。
更重要的是,這個(gè)counter變量是相互獨(dú)立的。比如,當(dāng)我們創(chuàng)建2個(gè)計(jì)數(shù)器時(shí),每個(gè)計(jì)數(shù)器都會(huì)創(chuàng)建一個(gè)新的counter變量:
// 每個(gè)計(jì)數(shù)器都會(huì)從1開(kāi)始計(jì)數(shù)
var counter1 = createCounter();
counter1(); // Number of events: 1
counter1(); // Number of events: 2
// 第1個(gè)計(jì)數(shù)器不會(huì)影響第2個(gè)計(jì)數(shù)器
var counter2 = createCounter();
counter2(); // Number of events: 1
// 第2個(gè)計(jì)數(shù)器不會(huì)影響第1個(gè)計(jì)數(shù)器
counter1(); // Number of events: 3
多個(gè)計(jì)數(shù)器的輸出信息都是“Number of events: N”,這樣容易混淆。如果可以為每個(gè)計(jì)數(shù)器命名,則更加方便:
var catCounter = createCounter("cats");
var dogCounter = createCounter("dogs");
catCounter(); // Number of cats: 1
catCounter(); // Number of cats: 2
dogCounter(); // Number of dogs: 1
通過(guò)給createCounter傳遞一個(gè)新的counterName參數(shù),可以很容易地做到這一點(diǎn):
function createCounter(counterName)
{
var counter = 0;
function increment()
{
counter = counter + 1;
console.log("Number of " + counterName + ": " + counter);
}
return increment;
}
這樣,createCounter()函數(shù)返回的計(jì)數(shù)器將同時(shí)記住兩個(gè)局部變量:counterName與counter。
按照之前的實(shí)現(xiàn)方式,我們通過(guò)調(diào)用createCounter()函數(shù)可以返回一個(gè)計(jì)數(shù)器,直接調(diào)用返回的計(jì)數(shù)器就可以加1,這樣做并不直觀。如果可以如下調(diào)用將更好:
var dogCounter = createCounter("dogs");
dogCounter.increment(); // Number of dogs: 1
實(shí)現(xiàn)代碼:
function createCounter(counterName)
{
var counter = 0;
function increment()
{
counter = counter + 1;
console.log("Number of " + counterName + ": " + counter);
};
return { increment : increment };
}
可知,以上的代碼返回了一個(gè)對(duì)象,這個(gè)對(duì)象包含了一個(gè)increment方法。
現(xiàn)在,我們可以給計(jì)數(shù)器添加一個(gè)decrement()方法
function createCounter(counterName)
{
var counter = 0;
function increment()
{
counter = counter + 1;
console.log("Number of " + counterName + ": " + counter);
};
function decrement()
{
counter = counter - 1;
console.log("Number of " + counterName + ": " + counter);
};
return {
increment : increment,
decrement : decrement
};
}
var dogsCounter = createCounter("dogs");
dogsCounter.increment(); // Number of dogs: 1
dogsCounter.increment(); // Number of dogs: 2
dogsCounter.decrement(); // Number of dogs: 1
前面的代碼有兩行重復(fù)的代碼,即console.log語(yǔ)句。因此,我們可以創(chuàng)建一個(gè)display()方法用于打印counter的值:
function createCounter(counterName)
{
var counter = 0;
function display()
{
console.log("Number of " + counterName + ": " + counter);
}
function increment()
{
counter = counter + 1;
display();
};
function decrement()
{
counter = counter - 1;
display();
};
return {
increment : increment,
decrement : decrement
};
}
var dogsCounter = createCounter("dogs");
dogsCounter.increment(); // Number of dogs: 1
dogsCounter.increment(); // Number of dogs: 2
dogsCounter.decrement(); // Number of dogs: 1
看起來(lái),display()函數(shù)與increment()函數(shù)以及decrement()函數(shù)差不多,但是其實(shí)它們很不一樣。我們并沒(méi)有將display()函數(shù)添加到返回的對(duì)象中,這就意味著以下代碼會(huì)出錯(cuò):
var dogsCounter = createCounter("dogs");
dogsCounter.display(); // ERROR !!!
這時(shí),display()相當(dāng)于一個(gè)私有方法,我們只能在createCounter()函數(shù)內(nèi)使用它。
如果你接觸過(guò)面向?qū)ο缶幊?OOP),則應(yīng)該不難發(fā)現(xiàn)本文中所涉及的內(nèi)容與OOP中的類、對(duì)象、對(duì)象屬性、共有方法與私有方法等概念非常相似。
閉包,與OOP相似,就是把數(shù)據(jù)和操作數(shù)據(jù)的方法綁定起來(lái)。因此,在需要OOP的時(shí)候,就可以使用閉包來(lái)實(shí)現(xiàn)。
閉包(Closure)是JavaScript一個(gè)非常棒的特性。掌握它,我們可以從容應(yīng)對(duì)一些常見(jiàn)的編程需求。
Fundebug專注于JavaScript、微信小程序、微信小游戲、支付寶小程序、React Native、Node.js和Java實(shí)時(shí)BUG監(jiān)控。 自從2016年雙十一正式上線,F(xiàn)undebug累計(jì)處理了7億+錯(cuò)誤事件,得到了Google、360、金山軟件、百姓網(wǎng)等眾多知名用戶的認(rèn)可。歡迎免費(fèi)試用!
轉(zhuǎn)載時(shí)請(qǐng)注明作者Fundebug以及本文地址:
https://blog.fundebug.com/2017/07/31/javascript-closure/
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。
當(dāng)前標(biāo)題:解密JavaScript閉包-創(chuàng)新互聯(lián)
網(wǎng)頁(yè)鏈接:http://aaarwkj.com/article2/coejoc.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供云服務(wù)器、品牌網(wǎng)站制作、靜態(tài)網(wǎng)站、網(wǎng)站排名、App開(kāi)發(fā)、手機(jī)網(wǎng)站建設(shè)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容