JavaScript中同名標識符優先級詳解

 

    一,局部變量先使用後聲明,不影響外部同名變量

  1. var x = 1; // --> 外部變量x  
  2. function fn(){  
  3.     alert(x);  // --> undefined 局部變量x先使用  
  4.     var x = 2; // 後聲明且賦值  
  5. }  
  6. fn();  
  7. alert(x); // --> 1<br> 

第一點,函數fn內第一句輸出x,x是在第二句才定義的。這在JS中是允許的,這裏的允許是指不會出現語法錯誤程序可以運行。

但在其它語言如C,Java中卻是不允許的。變量必須先聲明後使用,如

  1. public class Test {  
  2.     public static void main(String[] args) {  
  3.         System.out.println(x); // 先使用  
  4.         int x = 10; // 後聲明  
  5.     }  

Java中編譯器會提示錯誤,程序無法運行。

第二點,函數fn內的局部變量x不會影響到外部的變量x。即fn內alert輸出不是1,而是undefined。

二,形參優先級高於函數名

  1. function fn(fn){  
  2.     alert(fn);  
  3. }  
  4. fn('hello'); // --> "hello" 

可以看到函數名和形參同名都是fn,輸出的是字符串"hello",卻不是函數fn的函數體(fn.toString())。

三,形參優先級高於arguments

  1. function fn(arguments){  
  2.     alert(arguments);  
  3. }  
  4. fn('hello'); // --> "hello"<br> 

arguments對象可以直接在函數內使用,是語言本身提供的一個 特殊標識符 。

這裏剛好將形參聲明成與其同名。輸出可以看到是"hello"而非"[object Object]",即形參arguments覆蓋了語言本身提供的真正的arguments。

四,形參優先級高於只聲明卻未賦值的局部變量

  1. function fn(a){  
  2.     var a;  
  3.     alert(a);  
  4. }  
  5. fn('hello'); // --> "hello" 

函數fn形參爲a,函數內第一句僅聲明局部變量a,卻並未賦值。從輸出結果是"hello"而非undefined可以看出形參a優先級高於僅聲明卻未賦值的局部變量a。

五,聲明且賦值的局部變量優先級高於形參

  1. function fn(a){  
  2.     var a = 1;  
  3.     alert(a);  
  4. }  
  5. fn('hello'); // --> "1" 

函數fn形參爲a,函數內第一句僅聲明局部變量a,賦值爲1。從輸出結果是"1"而非"hello"可以看出聲明且賦值的局部變量a優先級高於形參a。

六,形參賦值給同名局部變量時

  1. function fn(a){  
  2.     var aa = a;  
  3.     alert(a);  
  4. }  
  5. fn('hello'); 

暫不運行,猜測下結果。如果按照第五點:聲明且賦值的局部變量優先級高於形參。那麼a將是undefined。但實際上a是"hello",即右a是形參a,左a纔是局部變量a。

JavaScript中同名標識符優先級

這裏的兩個a互不干擾,誰也沒覆蓋誰。這與剛剛說的賦值的局部變量優先級高於形參又矛盾了。但引擎這樣做的確是我們想要的,因爲並不希望var a = a後a是undefined。

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章