Javascript:必須知道的Javascript知識點之“作用域鏈”


代碼示例

 1 var xxxVar1 = 1;
 2 var outer = function(){ 
 3    var xxxVar2 = 2;
 4 
 5    var results = [];
 6    
 7    for(var i = 0; i< 3; i++)
 8    {
 9       var inner = function(){
10          var xxxVar3 = 3;
11          return xxxVar3 + xxxVar2 +xxxVar1 + i;
12       }
13       results .push(inner);
14    }
15 
16    return results;
17 }
18 
19 var xxxVar1 = 100;
20 var xxxVar2 = 200;
21 var xxxVar3 = 300;
22 var results = outer();
23 results[0]();
24 results[1]();
25 results[2]();

執行結果

發生了什么事情

很多人都可能知道上例的執行結果,但是不是所有人都明白為什么會是這樣的結果,包括我自己。

名詞解釋

  • 活動對象:一次函數調用開始的時候,javascript解釋器會收集函數體中的所有局部變量(以var形式聲明的變量),將這些局部變量存儲到一個稱為“活動對象”的對象里,所有變量都初始為undefined。
    代碼示例
    1 var fun = function(){
    2    alert(name);
    3    var name = '段光偉';
    4 }
    當執行這個函數時候時(fun()),函數體還沒執行到,當前的活動對象為[{ name: undefined }],因此fun()執行的結果為:
  • 函數的[scope]屬性:每個函數在定義的時候(生成函數實例的時候)都會分配一個[scope]屬性,這個屬性指向的當前的“作用域鏈”。這個屬性開發人員是訪問不到的,只有javascript能訪問。
  • 作用域鏈:當函數調用時,javascript引擎會維護一個這次調用的作用域鏈,這個作用域鏈條是函數的[scope]指向的作用域鏈加上函數調用時的活動對象,形式如[ 活動對象, 函數定義時的作用域鏈條]。
    代碼示例
     1 var a = 1;
     2 //步驟1:[ { a: 1, outer: undefined } ]
     3 
     4 var outer = function(){
     5    //步驟3:[ { b: undefined, inner: undefined } ,{ a: 1, outer: function } ]
     6    var b = 2;
     7    var inner = function(){
     8       //步驟5:[ {}, { b: 2, inner: function } ,{ a: 1, outer: function } ]
     9       return a + b;
    10    }
    11 
    12    //步驟4:[ { b: 2, inner: function } ,{ a: 1, outer: function } ]
    13    return inner();
    14 }
    15 
    16 //步驟2:[ { a: 1, outer: function } ]
    17 outer();

 作用域鏈規則

規則1

javascript一般運行在一定的宿主中,每個宿主都會提供一個“全局對象”,或者叫“全局活動對象”,這個全局對象是所有作用域鏈的根節點。

規則2

 “取值操作”(如:alert(xxxVar))的規則是,沿着作用域鏈依次查找名稱為“xxxVar”的變量,返回第一個找到的值,如果找不到就拋出異常(ReferenceError: xxxVar is not defined)。

規則3

 “賦值操作”(如:xxxVar = '段光偉')的規則是,沿着作用域鏈依次查找名稱為“xxxVar”的變量,覆蓋第一個找到的值,如果找不到就將“xxxVar”添加到全局對象中。

備注

“閉包”這個概念就是通過“作用域鏈”實現的,而C#是通過編譯器實現的,.NET並不支持。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM