欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

解密JavaScript閉包

譯者按:從最簡單的計數(shù)器開始,按照需求對代碼一步步優(yōu)化,我們可以領(lǐng)會閉包的神奇之處。

讓客戶滿意是我們工作的目標(biāo),不斷超越客戶的期望值來自于我們對這個行業(yè)的熱愛。我們立志把好的技術(shù)通過有效、簡單的方式提供給客戶,將通過不懈努力成為客戶在信息化領(lǐng)域值得信任、有價值的長期合作伙伴,公司提供的服務(wù)項目有:主機域名、網(wǎng)絡(luò)空間、營銷軟件、網(wǎng)站建設(shè)、蘭山網(wǎng)站維護、網(wǎng)站推廣。

  • 原文: Closures are not magic

  • 譯者: Fundebug
    *為了保證可讀性,本文采用意譯而非直譯。另外,本文版權(quán)歸原作者所有,翻譯僅用于學(xué)習(xí)。**

對于JavaScript新手來說,閉包(Closures)是一個很神奇的東西。這篇博客將通過一個非常淺顯的代碼示例來解釋閉包。

計數(shù)器

我們的目標(biāo)是實現(xiàn)一個計數(shù)器,它的效果如下:

increment();  // Number of events: 1
increment();  // Number of events: 2
increment();  // Number of events: 3

可知,每次執(zhí)行increment()都會輸出"Number of events: N",且N每次都會加1。

這個計數(shù)器最直觀的實現(xiàn)方式如下:

var counter = 0;

function increment() 
{
  counter = counter + 1;
  console.log("Number of events: " + counter);
}

多個計數(shù)器

以上的代碼非常簡單。但是,當(dāng)我們需要第二個計數(shù)器時,就會遇到問題了。當(dāng)然,我們可以實現(xiàn)兩個重復(fù)的計數(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)我們需要更多計數(shù)器時,使用這種方法將不太現(xiàn)實。這時,就需要神奇的閉包了。

使用閉包實現(xiàn)計數(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)建了兩個獨立的計數(shù)器counter1counter2,分別進行計數(shù),互不干撓。代碼看著有點奇怪,我們不妨拆分起來分析。

首先,我們來看看createCounter

  • 創(chuàng)建了一個局部變量counter
  • 創(chuàng)建了一個局部函數(shù)increment(),它可以對counter變量進行加1操作。
  • 將局部函數(shù)increment()返回。注意,返回的是函數(shù)本身,而不是函數(shù)調(diào)用的結(jié)果。

看起來,createCounter()函數(shù)與我們最初定義的計數(shù)器非常相似。唯一的不同點在于:createCounter()將計數(shù)器封裝在一個函數(shù)內(nèi),于是我們將它稱作閉包。

難以理解的一點在于,當(dāng)我們使用createCounter()函數(shù)創(chuàng)建計數(shù)器時,實際上創(chuàng)建了一個新的函數(shù):

// fancyNewCounter是一個新創(chuàng)建的函數(shù)
var fancyNewCounter = createCounter();

閉包的神奇之處在于。每次使用createCounter()函數(shù)創(chuàng)建計數(shù)器increment時,都會創(chuàng)建一個對應(yīng)的counter變量。并且,返回的increment函數(shù)會始終記住counter變量。

更重要的是,這個counter變量是相互獨立的。比如,當(dāng)我們創(chuàng)建2個計數(shù)器時,每個計數(shù)器都會創(chuàng)建一個新的counter變量:

// 每個計數(shù)器都會從1開始計數(shù)
var counter1 = createCounter();
counter1(); // Number of events: 1
counter1(); // Number of events: 2

// 第1個計數(shù)器不會影響第2個計數(shù)器
var counter2 = createCounter();
counter2(); // Number of events: 1

// 第2個計數(shù)器不會影響第1個計數(shù)器
counter1(); // Number of events: 3

為計數(shù)器命名

多個計數(shù)器的輸出信息都是“Number of events: N”,這樣容易混淆。如果可以為每個計數(shù)器命名,則更加方便:

var catCounter = createCounter("cats");
var dogCounter = createCounter("dogs");

catCounter(); // Number of cats: 1
catCounter(); // Number of cats: 2
dogCounter(); // Number of dogs: 1

通過給createCounter傳遞一個新的counterName參數(shù),可以很容易地做到這一點:

function createCounter(counterName) 
{
  var counter = 0;

  function increment() 
  {
    counter = counter + 1;
    console.log("Number of " + counterName + ": " + counter);
  }

  return increment;
}

這樣,createCounter()函數(shù)返回的計數(shù)器將同時記住兩個局部變量:counterNamecounter

優(yōu)化計數(shù)器調(diào)用方式

按照之前的實現(xiàn)方式,我們通過調(diào)用createCounter()函數(shù)可以返回一個計數(shù)器,直接調(diào)用返回的計數(shù)器就可以加1,這樣做并不直觀。如果可以如下調(diào)用將更好:

var dogCounter = createCounter("dogs");
dogCounter.increment(); // Number of dogs: 1

實現(xiàn)代碼:

function createCounter(counterName) 
{
  var counter = 0;

  function increment() 
  {
    counter = counter + 1;
    console.log("Number of " + counterName + ": " + counter);
  };

  return { increment : increment };
}

可知,以上的代碼返回了一個對象,這個對象包含了一個increment方法。

添加decrement方法

現(xiàn)在,我們可以給計數(shù)器添加一個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語句。因此,我們可以創(chuàng)建一個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

看起來,display()函數(shù)與increment()函數(shù)以及decrement()函數(shù)差不多,但是其實它們很不一樣。我們并沒有將display()函數(shù)添加到返回的對象中,這就意味著以下代碼會出錯:

var dogsCounter = createCounter("dogs");
dogsCounter.display(); // ERROR !!!

這時,display()相當(dāng)于一個私有方法,我們只能在createCounter()函數(shù)內(nèi)使用它。

閉包與面向?qū)ο缶幊?/h4>

如果你接觸過面向?qū)ο缶幊?OOP),則應(yīng)該不難發(fā)現(xiàn)本文中所涉及的內(nèi)容與OOP中的、對象、對象屬性、共有方法私有方法等概念非常相似。

閉包,與OOP相似,就是把數(shù)據(jù)和操作數(shù)據(jù)的方法綁定起來。因此,在需要OOP的時候,就可以使用閉包來實現(xiàn)。

總結(jié)

閉包(Closure)是JavaScript一個非常棒的特性。掌握它,我們可以從容應(yīng)對一些常見的編程需求。

關(guān)于Fundebug

Fundebug專注于JavaScript、微信小程序、微信小游戲、支付寶小程序、React Native、Node.js和Java實時BUG監(jiān)控。 自從2016年雙十一正式上線,F(xiàn)undebug累計處理了7億+錯誤事件,得到了Google、360、金山軟件、百姓網(wǎng)等眾多知名用戶的認可。歡迎免費試用!

解密JavaScript閉包

版權(quán)聲明

轉(zhuǎn)載時請注明作者Fundebug以及本文地址:
https://blog.fundebug.com/2017/07/31/javascript-closure/

本文標(biāo)題:解密JavaScript閉包
網(wǎng)站路徑:http://chinadenli.net/article24/gsggce.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供虛擬主機、網(wǎng)站建設(shè)、軟件開發(fā)、網(wǎng)站設(shè)計公司、定制開發(fā)、小程序開發(fā)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都網(wǎng)站建設(shè)公司