詳解JAVA字串型別switch的底層原理
阿新 • • 發佈:2019-09-11
基礎
我們現在使用的Java的版本,基本上是都支援String型別的。當然除了String型別,還有int、char、byte、short、enum等等也都是支援的。然而在其底部實現中,還是基於 整型的,也就是int、byte、short這些型別。
我們先來看一下int的一個簡單例子,主要部分原始碼
public static void main(String [] args){ int n = 2; switch (n){ case 1: break; case 2: break; case 3: break; default: } }
再使用javac命令編譯,javap命令反編譯之後得到如下關鍵部分位元組碼:
0: iconst_2 1: istore_1 2: iload_1 3: tableswitch { // 1 to 3 1: 28 2: 31 3: 34 default: 37 } 28: goto 37 31: goto 37 34: goto 37 37: return
看不懂的話可以 點選這裡 檢視參考對照表。
當然懶得看的話我們也可以直接把class檔案反編譯成原始碼,可以直接將class檔案拖進IDEA,得到如下程式碼:
public static void main(String[] var0) {
byte var1 = 2;
switch(var1) {
case 1:
case 2:
case 3:
default:
}
}
這裡總的來說和原始碼變化不大,只是將int型別都轉化為了byte型別。這裡轉化的原因,在於我們最初的case裡面的值剛好在byte的範圍之內。如果case的值稍微大點,它可能就會轉化為short型別,再大點,就直接是int型別了。需要注意的是switch裡面不支援float、long這些型別。
String型別講解
有了上文的理解之後,下面應該會簡單許多。
同樣的,還是先上原始碼
public static void main(String [] args){
String str = "sdf";
switch (str){
case "aaa":
break;
case "ccc":
break;
case "bbb":
break;
default:
}
}
然後編譯之後丟進IDEA反編譯得到反編譯的程式碼
public static void main(String[] var0) {
String var1 = "sdf";
byte var3 = -1;
switch(var1.hashCode()) {
case 96321:
if (var1.equals("aaa")) {
var3 = 0;
}
break;
case 97314:
if (var1.equals("bbb")) {
var3 = 2;
}
break;
case 98307:
if (var1.equals("ccc")) {
var3 = 1;
}
}
switch(var3) {
case 1:
case 0:
case 2:
default:
}
}
可以看到,String型別的switch,轉換為了字串的雜湊比較,而其雜湊返回的正是int型別。hash相同的情況再通過equals方法對比字串的值,因此引進區域性變數var3,是很有必要的