JavaScript這門語言的類型系統從來沒有它表面看起來的那樣和善,雖然比起Java、C#等一眾強類型語言,它的弱類型使用起來似乎是如此便利,但正因為它極高的自由度,所以才會衍生出令人摸不著頭腦的荒誕行為。
舉個例子,雖然我們都知道一個包含內容的字符串會被認為是“真值 Truthy”(因為除了空字符串之外任何字符串在JS里都被認為真值),但當你做如下比較的時候,你會得到一個驚掉下巴的結果
const a = "18";
const b = true;
a == b // false
什么鬼,一個被通常理解成真值的值,竟然無法與布爾真值松散相等?
為了能撥開JavaScript類型的迷霧,讓頭鐵的我們一點一點理順JavaScript整個類型系統的工作邏輯。
讀者可以根據自己對JS類型系統的掌握程度,選擇性的閱讀這篇博客
類型基礎
JavaScript有以下八大類型,除了object
類型,其他都為基本類型
number
string
boolean
null
undefined
object
symbol
bigint
他們的類型都可以直接被typeof
識別,特例是
雖然你可能已經為這種特例所不解,但其實這才剛開始,大的還在后面
類型轉換
轉換為數字
JavaScript內部有一套抽象的數字轉換機制叫ToNumber
,這套機制在隱式轉換或者部分顯式轉換其他類型值到數字時會被調用。雖然你可能會被惡心到,但我還是要向你介紹這套機制的規則為
ToNumber
的轉換規則會在以下情況下使用,當然這些情況也可以稱作轉換數字的“技巧”,看你怎么理解它了:
Number(value)
+ value
Math.floor(value)
value * x
,將一個值做乘法運算
沒錯,Math.floor(true)
、+ true
、true * 1
都等于1,是不是覺得很荒誕?
但'5' + 3
或者5 + '3'
結果都是字符串'53'
,因為+
在有兩個操作值,且其中一個為字符串時,會直接做字符串拼接。所以雖然+ '3'
結果為數字3,但5 + '3'
的結果不是8
parseInt()
或parseFloat()
只接受string
類型,所以轉換規則與ToNumber
轉換機制下的string
類型情況相似,但是在處理字符時采取從左到右的掃描直到失敗為止的方法,所以parseInt("123hello")
結果為123
轉換為布爾
JavaScript還有一套針對布爾類型的抽象轉換機制叫ToBoolean
。因為對于前端邏輯編寫來講,判斷一個值是否為真實在太重要了,JavaScript里變量像薛定諤的貓一樣,處于存在與不存在、真和假的中間態,所以我們JS開發者都有一個奇怪的腦回路,當看到一個字面量值的時候就開始評估它是“真值”(Trusy Value)還是“假值”(Falsy Value)。
可是,與物理學里薛定諤的貓現象相反,JavaScript里對真假值的定義其實很簡單,以下的值均為假值:
undefined
null
false
+0
、-0
、0n
和NaN
""
其他均為真值,任何非空字符串、非0數字、對象都是真值。
轉換其他類型值為布爾類型的方法:
!!value
Boolean(value)
!
,可將值轉換為布爾類型,但是真假結果相反
會隱式的將其他類型值轉換為布爾的情況:
恭喜你勇士,讀到這里就代表馬上你就能知道為什么"18" != true
了?。。?/p>
等價性
我們都知道,JS當中有松散判斷的弱等價==
,和嚴格判斷的強等價===
兩種判斷等價方式。強等價要求兩個操作值必須為同一類型,且值本身也相等,其行為非常容易預測。弱等價在比較值是否相等前會嘗試做一些類型轉換,盡可能的讓可能為不同類型的兩個值變得可以判斷。
造成文章開頭神奇判斷結果的原因就在弱等價==
時的類型轉換策略上,聽我將弱等價的轉換規則為你娓娓道來
所以,"18" == true
進行比較時
由于布爾值的比較規則為將布爾值轉換為數字,ToNumber(true)
結果為數字1
,表達式變為"18" == 1
又因為字符串與數字比較時字符串應轉換為數字,則ToNumber("18")
結果為數字18
最后表達式實際比較18 == 1
,結果為假false
所以沒錯,以下表達式均為真:"1" == true
,"0" == false
,false == ""
結語
了解這個例子后你可能會對JavaScript語言行為設計混亂表達不滿,但是上述邊界條件可以通過引入TypeScript,避免弱等價判斷轉而使用嚴格等價,在隱式類型轉換之前使用可預測行為的轉換方法對值提前進行處理。
轉自博客園 https://www.cnblogs.com/camwang/p/18242665 作者CamWang
該文章在 2024/6/14 8:46:25 編輯過