JavaScript 閉包
JavaScript 閉包
什么是閉包
閉包,官方對閉包的解釋是:一個擁有許多變量和綁定了這些變量的環(huán)境的表達式(通常是一個函數(shù)),因而這些變量也是該表達式的一部分。閉包的特點:
1. 作為一個函數(shù)變量的一個引用,當函數(shù)返回時,其處于激活狀態(tài)。
2. 一個閉包就是當一個函數(shù)返回時,一個沒有釋放資源的棧區(qū)。
簡單的說,Javascript允許使用內(nèi)部函數(shù)---即函數(shù)定義和函數(shù)表達式位于另一個函數(shù)的函數(shù)體內(nèi)。而且,這些內(nèi)部函數(shù)可以訪問它們所在的外部函數(shù)中聲明的所有局部變量、參數(shù)和聲明的其他內(nèi)部函數(shù)。當其中一個這樣的內(nèi)部函數(shù)在包含它們的外部函數(shù)之外被調(diào)用時,就會形成閉包。
2、閉包的幾種寫法和用法
首先要明白,在JS中一切都是對象,函數(shù)是對象的一種。下面先來看一下閉包的5種寫法,簡單理解一下什么是閉包。后面會具體解釋。
//第1種寫法 ?
function Circle(r) { this.r = r; } Circle.PI = 3.14159; Circle.prototype.area = function() { return Circle.PI * this.r * this.r; } var c = new Circle(1.0); alert(c.area());
這種寫法沒什么特別的,只是給函數(shù)添加一些屬性。
//第2種寫法 ?
var Circle = function() { var obj = new Object(); obj.PI = 3.14159; obj.area = function( r ) { return this.PI * r * r; } return obj; } var c = new Circle(); alert( c.area( 1.0 ) );
這種寫法是聲明一個變量,將一個函數(shù)當作值賦給變量。
//第3種寫法 ?
var Circle = new Object(); Circle.PI = 3.14159; Circle.Area = function( r ) { return this.PI * r * r; } alert( Circle.Area( 1.0 ) );
這種方法最好理解,就是new 一個對象,然后給對象添加屬性和方法。
//第4種寫法 ?
var Circle={ "PI":3.14159, "area":function(r){ return this.PI * r * r; } }; alert( Circle.area(1.0) );
這種方法使用較多,也最為方便。var obj = {}就是聲明一個空的對象。
//第5種寫法 ?
var Circle = new Function("this.PI = 3.14159;this.area = function( r ) {return r*r*this.PI;}"); ???
alert( (new Circle()).area(1.0) ); ?
說實話,這種寫法我是沒用過,大家可以參考一下。?
總的來說,上面幾種方法,第2中和第4中較為常見,大家可以根據(jù)習慣選擇。?
上面代碼中出現(xiàn)了JS中常用的Prototype,那么Prototype有什么用呢?下面我們來看一下:
var dom = function(){ }; dom.Show = function(){ alert("Show Message"); }; dom.prototype.Display = function(){ alert("Property Message"); }; dom.Display(); //error dom.Show(); var d = new dom(); d.Display(); d.Show(); //error
變量生命周期
全局變量的作用域是全局性的,即在整個JavaScript程序中,全局變量處處都在。
而在函數(shù)內(nèi)部聲明的變量,只在函數(shù)內(nèi)部起作用。這些變量是局部變量,作用域是局部性的;函數(shù)的參數(shù)也是局部性的,只在函數(shù)內(nèi)部起作用。
計數(shù)器困境
設(shè)想下如果你想統(tǒng)計一些數(shù)值,且該計數(shù)器在所有函數(shù)中都是可用的。
你可以使用全局變量,函數(shù)設(shè)置計數(shù)器遞增:
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> </head> <body> <p>全局變量計數(shù)。</p> <button type="button" onclick="myFunction()">計數(shù)!</button> <p id="demo">0</p> <script> var counter = 0; function add() { return counter += 1; } function myFunction(){ document.getElementById("demo").innerHTML = add(); } </script> </body> </html>
計數(shù)器數(shù)值在執(zhí)行 add() 函數(shù)時發(fā)生變化。
但問題來了,頁面上的任何腳本都能改變計數(shù)器,即便沒有調(diào)用 add() 函數(shù)。
如果我在函數(shù)內(nèi)聲明計數(shù)器,如果沒有調(diào)用函數(shù)將無法修改計數(shù)器的值:
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> </head> <body> <p>局部變量計數(shù)。</p> <button type="button" onclick="myFunction()">計數(shù)!</button> <p id="demo">0</p> <script> function add() { var counter = 0; return counter += 1; } function myFunction(){ document.getElementById("demo").innerHTML = add(); } </script> </body> </html>
以上代碼將無法正確輸出,每次我調(diào)用 add() 函數(shù),計數(shù)器都會設(shè)置為 1。
JavaScript 內(nèi)嵌函數(shù)可以解決該問題。
JavaScript 內(nèi)嵌函數(shù)
所有函數(shù)都能訪問全局變量。??
實際上,在 JavaScript 中,所有函數(shù)都能訪問它們上一層的作用域。
JavaScript 支持嵌套函數(shù)。嵌套函數(shù)可以訪問上一層的函數(shù)變量。
該實例中,內(nèi)嵌函數(shù)?plus()?可以訪問父函數(shù)的?counter?變量:
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> </head> <body> <p>局部變量計數(shù)。</p> <p id="demo">0</p> <script> document.getElementById("demo").innerHTML = add(); function add() { var counter = 0; function plus() {counter += 1;} plus(); return counter; } </script> </body> </html>
如果我們能在外部訪問?plus()?函數(shù),這樣就能解決計數(shù)器的困境。
我們同樣需要確保?counter = 0?只執(zhí)行一次。
我們需要閉包。