這篇文章主要介紹了JavaScript變數作用域,小編覺得蠻不錯的,現在分享給大家,也給大家做個參考。一起跟著小編過來看看吧
在JavaScript中,用var
申明的變數其實是有作用域的。
如果一個變數在函數體內部申明,則該變數的作用域為整個函數體,在函數體外不可引用該變數:
'use strict'; function foo() { var x = 1; x = x + 1; } x = x + 2; // ReferenceError! 无法在函数体外引用变量x
如果兩個不同的函數各自申明了同一個變量,那麼該變數只在各自的函數體內運作。換句話說,不同函數內部的同名變數互相獨立,不影響:
'use strict'; function foo() { var x = 1; x = x + 1; } function bar() { var x = 'A'; x = x + 'B'; }
由於JavaScript的函數可以嵌套,此時,內部函數可以存取外部函數定義的變量,反過來則不行:
'use strict'; function foo() { var x = 1; function bar() { var y = x + 1; // bar可以访问foo的变量x! } var z = y + 1; // ReferenceError! foo不可以访问bar的变量y! }
如果內部函數和外部函數的變數名稱重名怎麼辦?
'use strict'; function foo() { var x = 1; function bar() { var x = 'A'; alert('x in bar() = ' + x); // 'A' } alert('x in foo() = ' + x); // 1 bar(); }
這表示JavaScript的函數在尋找變數時從自身函數定義開始,並從「內」向「外」尋找。如果內部函數定義了與外部函數重名的變量,則內部函數的變數將「屏蔽」外部函數的變數。
變數提升
JavaScript的函數定義有個特點,它會先掃描整個函數體的語句,把所有申明的變數「提升」到函數頂部:
'use strict'; function foo() { var x = 'Hello, ' + y; alert(x); var y = 'Bob'; } foo();
雖然是strict模式,但語句var x = 'Hello, ' + y;
並傳錯,原因是變數y
稍後申明了。但是alert
顯示Hello, undefined
,說明變數y
的值為undefined
。這正是因為JavaScript引擎自動提升了變數y
的聲明,但不會提升變數y
的賦值。
對於上述foo()
函數,JavaScript引擎看到的程式碼相當於:
##
function foo() { var y; // 提升变量y的申明 var x = 'Hello, ' + y; alert(x); y = 'Bob'; }由於JavaScript的這個怪異的“特性”,我們在函數內部
定義變數時,請嚴格遵守“在函數內部首先申明所有變數”這一規則。最常見的做法是用一個var申明函數內部用到的所有變數:
function foo() { var x = 1, // x初始化为1 y = x + 1, // y初始化为2 z, i; // z和i为undefined // 其他语句: for (i=0; i<100; i++) { ... } }
全域作用域
window,全域作用域的變數實際上被綁定到
window的一個屬性:
'use strict'; var course = 'Learn JavaScript'; alert(course); // 'Learn JavaScript' alert(window.course); // 'Learn JavaScript'因此,直接存取全域變數
course和存取
window.course是完全一樣的。
var foo = function () {}#定義的函數實際上也是一個全域變量,因此,頂層函數的定義也被視為一個全域變量,並綁定到
window物件:
'use strict'; function foo() { alert('foo'); } foo(); // 直接调用foo() window.foo(); // 通过window.foo()调用進一步大膽地猜測,我們每次直接呼叫的alert()函數其實也是window的變數:
'use strict'; window.alert('调用window.alert()'); // 把alert保存到另一个变量: var old_alert = window.alert; // 给alert赋一个新函数: window.alert = function () {} // 恢复alert: window.alert = old_alert; alert('又可以用alert()了!');這表示JavaScript其實只有一個全域作用域。任何變數(函數也視為變數),如果沒有在目前
函數作用域中找到,就會繼續往上查找,最後如果在全域作用域中也沒有找到,則報ReferenceError錯誤。
名字空間
window上,不同的JavaScript檔案如果使用了相同的全域變量,或定義了相同名字的頂層函數,都會造成命名衝突,而且很難被發現。
// 唯一的全局变量MYAPP: var MYAPP = {}; // 其他变量: MYAPP.name = 'myapp'; MYAPP.version = 1.0; // 其他函数: MYAPP.foo = function () { return 'foo'; };把自己的程式碼全部放入唯一的名字空間
MYAPP中,會大幅減少全域變數衝突的可能性。
局部作用域
for循環等語句區塊中是無法定義具有局部作用域的變數的:
'use strict'; function foo() { for (var i=0; i<100; i++) { // } i += 100; // 仍然可以引用变量i }為了解決區塊級作用域,ES6引入了新的關鍵字let,用let替代var可以申明一個區塊級作用域的變數:
'use strict'; function foo() { var sum = 0; for (let i=0; i<100; i++) { sum += i; } i += 1; // SyntaxError }
常數
#
由于var
和let
申明的是变量,如果要申明一个常量,在ES6之前是不行的,我们通常用全部大写的变量来表示“这是一个常量,不要修改它的值”:
var PI = 3.14;
ES6标准引入了新的关键字const来定义常量,const与let都具有块级作用域:
'use strict'; const PI = 3.14; PI = 3; // 某些浏览器不报错,但是无效果! PI; // 3.14
以上是介紹JavaScript變數作用域實例的詳細內容。更多資訊請關注PHP中文網其他相關文章!