switch
Java7开始,switch的参数可以是String类型了,这真的是一个很有用的改进,毕竟string还是挺常用的。到目前为止,switch支持的参数类型有:byte、short、int、char、String、enum。switch对各种类型参数的支持到底是怎么实现的呢?
byte、short、int
public class Main2 {
    public static void main(String[] args) {
        byte b = 1;
        switch (b) {
            case 0:
                System.out.println("0");
                break;
            case 1:
                System.out.println("1");
                break;
            default:
                System.out.println("other");
        }
    }
}
反编译:
public class Main2 {
    public Main2() {
    }
    public static void main(String[] args) {
        byte b = 1;
        switch(b) {
            case 0:
                System.out.println("0");
                break;
            case 1:
                System.out.println("1");
                break;
            default:
                System.out.println("other");
        }
    }
}
将byte换成short、int,编译后的结果与上面展示的基本相同。
char
public class Main2 {
    public static void main(String[] args) {
        char c = '0';
        switch (c) {
            case '0':
                System.out.println("0");
                break;
            case '1':
                System.out.println("1");
                break;
            default:
                System.out.println("other");
        }
    }
}
反编译:
public class Main2 {
    public Main2() {
    }
    public static void main(String[] args) {
        char c = 48;
        switch(c) {
            case 48:
                System.out.println("0");
                break;
            case 49:
                System.out.println("1");
                break;
            default:
                System.out.println("other");
        }
    }
}
从反编译的结果看,对char类型进行比较的时候,实际比较的是字符对应的ASCⅡ码,编译器把switch中char类型的变量转换成int类型的变量。
String
public class Main2 {
    public static void main(String[] args) {
        String s = "0";
        switch (s) {
            case "0":
                System.out.println("0");
                break;
            case "1":
                System.out.println("1");
                break;
            default:
                System.out.println("other");
        }
    }
}
反编译:
public class Main2 {
    public Main2() {
    }
    public static void main(String[] args) {
        String s = "0";
        byte var3 = -1;
        switch(s.hashCode()) {
            case 48:
                if (s.equals("0")) {
                    var3 = 0;
                }
                break;
            case 49:
                if (s.equals("1")) {
                    var3 = 1;
                }
        }
        switch(var3) {
            case 0:
                System.out.println("0");
                break;
            case 1:
                System.out.println("1");
                break;
            default:
                System.out.println("other");
         }
    }
}
从反编译的结果看,switch借助hashCode()和equals()实现了对String的支持——先使用hashCode进行初步判断,然后使用equal()进行二次校验(由于哈希冲突的存在,这个二次校验是必要的)。
enum
public enum ColorEnum {
    RED, YELLOW, BLUE, GREED
}
public class Main {
    public static void main(String[] args) {
        ColorEnum color = ColorEnum.YELLOW;
        switch (color) {
            case RED:
                System.out.println("red");
                break;
            case YELLOW:
                System.out.println("yellow");
                break;
            default:
                System.out.println("other");
        }
    }
}
反编译:
/*
 * Decompiled with CFR 0.149.
 */
package com.learn.java;
public final class ColorEnum
extends Enum<ColorEnum> {
    public static final /* enum */ ColorEnum RED = new ColorEnum("RED", 0);
    public static final /* enum */ ColorEnum YELLOW = new ColorEnum("YELLOW", 1);
    public static final /* enum */ ColorEnum BLUE = new ColorEnum("BLUE", 2);
    public static final /* enum */ ColorEnum GREED = new ColorEnum("GREED", 3);
    private static final /* synthetic */ ColorEnum[] $VALUES;
    public static ColorEnum[] values() {
        return (ColorEnum[])$VALUES.clone();
    }
    public static ColorEnum valueOf(String string) {
        return Enum.valueOf(ColorEnum.class, string);
    }
    private ColorEnum(String string, int n) {
        super(string, n);
    }
    static {
        $VALUES = new ColorEnum[]{RED, YELLOW, BLUE, GREED};
    }
}
/*
 * Decompiled with CFR 0.149.
 */
package com.learn.java;
import com.learn.java.ColorEnum;
public class Main {
    public static void main(String[] arrstring) {
        ColorEnum colorEnum = ColorEnum.YELLOW;
        switch (1.$SwitchMap$com$learn$java$ColorEnum[colorEnum.ordinal()]) {
            case 1: {
                System.out.println("red");
                break;
            }
            case 2: {
                System.out.println("yellow");
                break;
            }
            default: {
                System.out.println("other");
            }
        }
    }
}
上面的反编译结果表明,switch借助枚举类的序号实现对枚举类的支持。
总结
其实switch只支持整型,其他数据类型都是转换成整型后才使用的switch。
参考文献:https://hollischuang.github.io/toBeTopJavaer/#/basics/java-basic/switch-string
    地势坤,君子以厚德载物。
 
                    
                     
                    
                 
                    
                 
 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号