Java 中 switch 語句支持字符串類型的歷史和原理
想象一下這個場景:你正在寫一個水果識別的程序,用戶輸入「蘋果」就要執行特定邏輯。在 Java 7 之前,你只能這樣寫:
if (fruit.equals("蘋果")) {
// 處理蘋果
} else if (fruit.equals("香蕉")) {
// 處理香蕉
} else {
//... 其他水果
}
這種寫法就像超市收銀員每次結賬都要拆開包裹檢查商品,而 Java 7 之后的字符串 switch 相當于給每個商品貼了快速識別條形碼-程序員寫代碼更優雅。
以前為什么不能用字符串?
早期的 Java(Java 7 以前)switch 只能處理數字類數據(比如 int、char),因為這些類型底層可以直接用數字比對,簡單快速。而字符串是復雜對象,直接逐個字符對比太費時間,所以那時候只能用 if-else 處理字符串分支。
字符串的“身份證”——哈希碼(Hash Code)
每個字符串都有個唯一的哈希碼(可以理解成根據內容用數學公式算出來的身份證號):
- ? 比如 "蘋果" 的哈希碼是 12345,"香蕉" 的哈希碼是 67890
- ? 關鍵特性:內容相同的字符串哈希碼一定相同,不同內容大概率不同(極小概率重復,叫“哈希碰撞”)
switch 字符串操作
1. 存身份證號編譯器會把所有 case 后的字符串(比如 "蘋果"、"香蕉")的哈希碼存進一個數組里。
2. 先比身份證號運行到 switch 時,先計算輸入字符串的哈希碼(比如用戶輸入 "蘋果",算出 12345),然后去數組里快速匹配。
3. 再驗真身(防冒牌貨)哈希碼匹配后,還要用 equals() 方法確認字符串內容是否真的相同(防止極小概率的哈希碰撞)。
從本質來講,switch 對字符串的支持,其實也是 int 類型值的匹配。
舉個栗子 ??
String fruit = "蘋果";
switch(fruit) {
case "蘋果":
System.out.println("紅富士");
break;
case "香蕉":
System.out.println("芝麻蕉");
break;
default:
System.out.println("不認識");
}
在這個例子中,編譯器會先計算 "蘋果" 和 "香蕉" 的哈希碼,然后在運行時計算 fruit 的哈希碼。如果 fruit 的哈希碼和 "蘋果" 的哈希碼匹配,它會再用 equals 方法確認 fruit 是否真的是 "蘋果"。如果是,就輸出 "紅富士"。
編譯器的優化
為了讓 switch 語句更高效,編譯器還會做一些優化。比如,如果 case 標簽不多,編譯器可能會直接用一系列的 if-else 語句來代替 switch 語句。這樣做可以減少 hashCode 計算和 equals 方法調用的次數,從而提高性能。
性能考慮
雖然字符串類型的 switch 語句很方便,但它畢竟需要計算哈希碼 hashCode 和調用 equals 方法,所以在性能要求非常高的場合,可能不如用整數類型或枚舉類型來得快。比如,在一些對速度要求極高的游戲中,開發者可能會選擇用整數或枚舉類型來避免額外的性能開銷。switch 語句讓編程變得更加靈活,但也需要注意它的性能特點,合理選擇適用的場景。