1. 程式人生 > >詳解JAVA字串型別switch的底層原理

詳解JAVA字串型別switch的底層原理

基礎

我們現在使用的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,是很有必要的