在Java里switch语句支持哪些类型_Java分支结构使用说明

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:仅支持 byteshortcharint 及其对应的包装类(自动拆箱)
  • Java 7:新增 String —— 注意是引用相等(equals),不是 ==
  • Java 14(预览)→ Java 17(正式):支持 enum 类型(其实 Java 5 就支持 enum 常量作为 case,但完整 enum 类型在语义上更清晰)
  • Java 14 起引入 switch 表达式(带 ->yield),但类型限制和语句版一致
  • 至今不支持longfloatdoubleboolean 及任意自定义 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 编译后通常生成 tableswitchlookupswitch 字节码指令,它们都要求跳转键(key)能映射为 32 位有符号整数范围内的索引。

  • long 值范围远超 int,无法安全压缩进跳转表
  • 没有隐式截断机制(比如把 long x = 0x1_0000_0000L 当作 0 处理),那会破坏语义一致性
  • 替代方案明确:转成 int(需确认无精度丢失)、用 if-else、或封装进 Map / EnumMap

真正容易被忽略的是:哪怕你只用 long 的小数值(比如 1L、2L、3L),编译器也坚决拒绝——它看的是类型,不是实际值。