Java中switch编译后生成tableswitch或lookupswitch字节码:tableswitch适用于密集连续case值,时间复杂度O(1)但空间开销大;lookupswitch适用于稀疏分散case值,时间复杂度O(log n)但内存占用小。
Java中switch语句在编译后不会直接保留为高级语法,而是根据case值的分布特征,由javac编译器自动选择生成tableswitch或lookupswitch字节码指令——二者都是JVM为高效跳转设计的分支指令,但适用场景和性能表现不同。
当switch的case常量是相对紧凑、跨度小且基本连续(比如0,1,2,3,5,6,中间最多缺少数个)时,javac倾向于生成tableswitch。它底层用一个“跳转表”实现:JVM先检查switch表达式的值是否落在[low, high]范围内,若在,就直接以该值为索引查表获取对应branch offset;若超出,则跳转到default分支。
switch(n) { case 0: ... case 1: ... case 2: ... case 10: ... }(共11个case,最大最小差为10)大概率触发tableswitch当case值跨度大、不连续、数量不多(如case 100、case 1000、case 99999),javac会生成lookupswitch。它内部维护一个已排序的键值对列表(key=case值,value=branch offset),执行时JVM对该列表做二分查找,找到匹配key后跳转。
switch(x) { case 1: ... case 100: ... case 10000: ... } 编译后几乎必为lookupswitch用javap -c反编译class文件即可直观看到:
tableswitch关键字,后面跟着low、high及一串offset列表 → tableswitchlookupswitch关键字,后面是npairs(键值对数量)及成对的match: offset → lookupswitchString.hashCode()再套一层switch,最终仍落在这两种指令之一;枚举switch同理,本质是转为ordinal()后再分支不能。javac完全自主决策,开发者无法用语法或注解指定。但可通过调整case分布“引导”编译器倾向某一种:
)对二者都有良好优化,日常开发无需为此微调逻辑,可读性和正确性优先基本上就这些。理解tableswitch与lookupswitch的区别,有助于读懂字节码、分析分支性能,也能在做底层优化或写编译器相关工具时少走弯路。