Java 5–6仅支持byte/short/char/int及包装类;Java 7新增String(需编译期常量,null抛NPE);Java 14起支持switch表达式(必须全覆盖、用->和yield);long等类型因JVM跳转表限制不支持。

switch 支持的类型从 Java 5 到 Java 14 的演进
Java 的 switch 语句不是一开始就能处理任意类型的——它的支持范围是逐步扩展的。理解当前 JDK 版本能用什么,直接决定你写不写得出编译通过的代码。
- Java 5–6:仅支持
byte、short、char、int及其对应的包装类(自动拆箱) - Java 7:新增
String—— 注意是引用相等(equals),不是== - Java 14(预览)→ Java 17(正式):支持
enum类型(其实 Java 5 就支持 enum 常量作为 case,但完整 enum 类型在语义上更清晰) - Java 14 起引入
switch表达式(带->和yield),但类型限制和语句版一致 -
至今不支持:
long、float、double、boolean及任意自定义 class(包括 record,除非显式转成 String 或 enum)
String 类型 switch 的实际约束和坑
虽然 Java 7 就加了 String 支持,但很多人忽略它底层仍依赖 hashCode() 和 equals(),且 case 值必须是编译期常量。
- case 中不能写
str.toUpperCase()这类运行时表达式,会报错constant string expression required - 空字符串
""是合法 case;但null传入会直接抛NullPointerException(不会进入任何 case) - 区分大小写:case
"GET"不匹配"get",没隐式转换 - 性能上,JVM 对 String switch 做了优化(类似查找表 + hashCode 分桶),一般比一连串
if (s.equals("a"))略快,但别指望替代 Map 查找
switch 表达式(Java 14+)怎么写才不报错
新式 switch 表达式要求**必须覆盖所有可能路径**,否则编译失败。这不是警告,是硬性检查。
String day = "MON";
int num = switch (day) {
case "MON", "TUE", "WED" -> 1;
case "THU", "FRI" -> 2;
case "SAT", "SUN" -> 3;
default -> -1; // 没这行?编译报错:'switch' expression must be complete
};
- 必须用
->(不是:),且每个分支结尾不能有break - 如果分支逻辑多于单表达式,要用
{}包裹,并用yield返回值:case "A" -> { System.out.println("hit"); yield 100; } - 对枚举使用时,如果 enum 加了新常量但没更新 switch,编译器会提示“missing case label”,强制你处理
为什么 long 不能用在 switch 里
这不是 JVM 实现偷懒,而是设计权衡:switch 编译后通常生成 tableswitch 或 lookupswitch 字节码指令,它们都要求跳转键(key)能映射为 32 位有符号整数范围内的索引。
立即学习“Java免费学习笔记(深入)”;
-
long值范围远超int,无法安全压缩进跳转表 - 没有隐式截断机制(比如把
long x = 0x1_0000_0000L当作0处理),那会破坏语义一致性 - 替代方案明确:转成
int(需确认无精度丢失)、用if-else、或封装进 Map / EnumMap
真正容易被忽略的是:哪怕你只用 long 的小数值(比如 1L、2L、3L),编译器也坚决拒绝——它看的是类型,不是实际值。










