詳解一下 javascript 中的比較

代碼1: 

  1. [] == [];  // false  
  2. [] === []; // false  
  3. {} == {};  // SyntaxError: Unexpected token ==  
  4. {} === {}; // SyntaxError: Unexpected token ===  

代碼2:

  1. var n0 = 123;  
  2. var n1 = new Number(123);  
  3. var n2 = new Number(123);  
  4. var n3 = Number(123);  
  5. var n4 = Number(123);  
  6.   
  7. n0 == n1;  // true  
  8. n0 == n3;  // true  
  9. n0 === n1; // false  
  10. n0 === n3; // true  
  11.   
  12. n1 == n2;  // false  
  13. n1 === n2; // false  
  14. n1 == n3;  // true  
  15. n1 === n3; // false  
  16.   
  17. n3 == n4;  // true  
  18. n3 === n4; // true  

 爲什麼會是這個結果? 

我們需要仔細的閱讀規範: http://ecmascript.cn/#203

11.9.3 抽象相等比較算法

比較運算 x==y, 其中 x 和 y 是值,產生 true 或者 false。這樣的比較按如下方式進行:

  1. 若 Type(x) 與 Type(y) 相同, 則
    1. 若 Type(x) 爲 Undefined, 返回 true
    2. 若 Type(x) 爲 Null, 返回 true
    3. 若 Type(x) 爲 Number, 則
      1. 若 x 爲 NaN, 返回 false
      2. 若 y 爲 NaN, 返回 false
      3. 若 x 與 y 爲相等數值, 返回 true
      4. 若 x 爲 +0 且 y 爲 −0, 返回 true
      5. 若 x 爲 −0 且 y 爲 +0, 返回 true
      6. 返回 false
    4. 若 Type(x) 爲 String, 則當 x 和 y 爲完全相同的字符序列(長度相等且相同字符在相同位置)時返回 true。 否則, 返回 false
    5. 若 Type(x) 爲 Boolean, 當 x 和 y 爲同爲 true 或者同爲 false 時返回 true。 否則, 返回false
    6. 當 x 和 y 爲引用同一對象時返回 true。否則,返回 false
  2. 若 x 爲 null 且 y 爲 undefined, 返回 true
  3. 若 x 爲 undefined 且 y 爲 null, 返回 true
  4. 若 Type(x) 爲 Number 且 Type(y) 爲 String, 返回 comparison x == ToNumber(y) 的結果。
  5. 若 Type(x) 爲 String 且 Type(y) 爲 Number
  6. 返回比較 ToNumber(x) == y 的結果。
  7. 若 Type(x) 爲 Boolean, 返回比較 ToNumber(x) == y 的結果。
  8. 若 Type(y) 爲 Boolean, 返回比較 x == ToNumber(y) 的結果。
  9. 若 Type(x) 爲 String 或 Number,且 Type(y) 爲 Object,返回比較 x == ToPrimitive(y) 的結果。
  10. 若 Type(x) 爲 Object 且 Type(y) 爲 String 或 Number, 返回比較 ToPrimitive(x) == y 的結果。
  11. 返回 false

注:按以上相等之定義:

  • 字符串比較可以按這種方式強制執行: "" + a == "" + b
  • 數值比較可以按這種方式強制執行: +a == +b
  • 布爾值比較可以按這種方式強制執行: !a == !b

注:等值比較操作保證以下不變:

  • A != B 等價於 !(A==B)
  • A == B 等價於 B == A,除了 A 與 B 的執行順序。

注:相等運算符不總是傳遞的。 例如,兩個不同的 String 對象,都表示相同的字符串值;== 運算符認爲每個 String 對象都與字符串值相等,但是兩個字符串對象互不相等。例如:

  • new String("a") == "a" 和 "a" == new String("a") 皆爲 true
  • new String("a") == new String("a") 爲 false

字符串比較使用的方式是簡單地檢測字符編碼單元序列是否相同。不會做更復雜的、基於語義的字符或者字符串相等的定義以及 Unicode 規範中定義的 collating order。所以 Unicode 標準中認爲相等的 String 值可能被檢測爲不等。實際上這一算法認爲兩個字符串已經是經過規範化的形式。

11.9.6 嚴格等於比較算法

比較 x===yx 和 y 爲值,需要產出 true 或 false。比較過程如下:

  1. 如果 Type(x) 與 Type(y) 的結果不一致,返回 false,否則
  2. 如果 Type(x) 結果爲 Undefined,返回 true
  3. 如果 Type(x) 結果爲 Null,返回 true
  4. 如果 Type(x) 結果爲 Number,則
    1. 如果 x 爲 NaN,返回 false
    2. 如果 y 爲 NaN,返回 false
    3. 如果 x 與 y 爲同一個數字,返回 true
    4. 如果 x 爲 +0y 爲 -0,返回 true
    5. 如果 x 爲 -0y 爲 +0,返回 true
    6. 返回 false
  5. 如果 Type(x) 結果爲 String,如果 x 與 y 爲完全相同的字符序列(相同的長度和相同的字符對應相同的位置),返回 true,否則,返回 false
  6. 如果 Type(x) 結果爲 Boolean,如果 x 與 y 都爲 true 或 false,則返回 true,否則,返回false
  7. 如果 x 和 y 引用到同一個 Object 對象,返回 true,否則,返回 false

注:此算法與 SameValue 算法在對待有符號的零和 NaN 上表現不同。

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