在javascript中,作用域是變量(對象、函數(shù))的可訪問范圍,是變量在腳本代碼中的可讀、寫的有效范圍;作用域可以控制變量的可見性和生命周期。
本教程操作環(huán)境:windows7系統(tǒng)、javascript1.8.5版、Dell G3電腦。
幾乎所有的語言都有作用域的概念,簡單的說,作用域(scope)就是變量的可訪問范圍,即作用域控制變量的可見性和生命周期。
在 JavaScript 中, 對象和函數(shù)同樣也是變量。
在 ECMAScript6 之前,變量的作用域主要分為全局作用域、局部作用域(也稱函數(shù)作用域)兩種;在 ECMAScript6 及其之后,變量的作用域主要分為全局作用域、局部作用域和塊級作用域這 3 種。
相應(yīng)作用域的變量分別稱為全局變量、局部變量和塊級變量。
-
全局變量聲明在所有函數(shù)之外;
-
局部變量是在函數(shù)體內(nèi)聲明的變量或者是函數(shù)的命名參數(shù);
-
塊級變量是在塊中聲明的變量,只在塊中有效。
變量的作用域跟聲明方式有很密切的關(guān)系。使用 var 聲明的變量的作用域有全局作用域和函數(shù)作用域,沒有塊級作用域;使用 let 和 const 聲明的變量有全局作用域、局部作用域和塊級作用域。
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> </head> <body> <script> var v1 = "JavaScript"; //全局變量 let v2 = "JScript"; //全局變量 let v3 = "Script"; //全局變量 scopeTest(); //調(diào)用函數(shù) function scopeTest() { var lv = "aaa"; //局部變量 var v1 = "bbb"; //局部變量 let v2 = "ccc"; //局部變量 if (true) { let lv = "123"; //塊級變量 console.log("塊內(nèi)輸出的lv = " + lv); //123 } console.log("函數(shù)體內(nèi)輸出的lv = " + lv); //aaa console.log("函數(shù)體內(nèi)輸出的v1 = " + v1); //bbb console.log("函數(shù)體內(nèi)輸出的v2 = " + v2); //ccc console.log("函數(shù)體內(nèi)輸出的v3 = " + v3); //Script //v4為全局變量,賦值在后面,因而值為undefined console.log("函數(shù)體內(nèi)輸出的v4 = " + v4); } var v4 = "VBScript"; //全局變量 console.log("函數(shù)體外輸出的lv = " + lv); //① 報(bào)ReferenceError錯誤 console.log("函數(shù)體外輸出的v1 = " + v1); //JavaScript console.log("函數(shù)體外輸出的v2 = " + v2); //JScript console.log("函數(shù)體外輸出的v3 = " + v3); //Script console.log("函數(shù)體外輸出的v3 = " + v4); //VBScript </script> </body> </html>
上述腳本代碼分別聲明了 4 個全局變量、3 個局部變量和 1 個塊級變量。在 scopeTest 函數(shù)體外,變量 v1、v2、v3 和 v4 為全局變量;在 scopeTest 函數(shù)體內(nèi),lv、v2是全局變量;在 if 判斷塊中,lv 是塊級變量。
我們看到,局部變量 v1 和 v2 與全局變量 v1 和 v2 同名,在 scopeTest 函數(shù)體內(nèi),局部變量 v1 和 v2 有效,因而在函數(shù)體這 2 個變量的輸出結(jié)果分別為“bbb”和“ccc”;在函數(shù)體外,全局變量 v1 和 v2 有效,因而在函數(shù)體外,這 2 個變量的輸出結(jié)果分別為“JavaScript”和“JScript”。
另外,塊級變量 lv 和局部變量 lv 同名,在 if 判斷塊中,塊級變量 lv 有效,因而在塊中輸出的結(jié)果為“123”,而在塊外,局部變量 lv 有效,lv 變量的輸出結(jié)果為“aaa”。
另外,全局變量 v3 和 v4 在函數(shù)體中沒有被覆蓋,因而輸出的是全局變量的值,所以 v3 在函數(shù)體內(nèi)和體外的輸出結(jié)果都為“Script”,而 v4 變量的賦值在函數(shù)調(diào)用的后面,因而在函數(shù)體中的 v4 輸出結(jié)果為“undefined”,而在函數(shù)體外的輸出是在聲明之后,所以結(jié)果為“VBScript”。lv 是局部變量,因而在函數(shù)體外訪問會報(bào)“ReferenceError”錯誤。
上述代碼在 Chrome 瀏覽器中運(yùn)行后,打開瀏覽器的控制臺,可以看到下圖所示的輸出結(jié)果。
28行代碼報(bào)錯的原因:
lv 變量為局部變量,離開函數(shù)后無效。將這行代碼注釋后再運(yùn)行,此時打開瀏覽器控制臺可看到:
【