JAVA字符串类型switch的底层原理详析
前言
switch语句是非常的基础的知识,掌握起来也不难掌握,语法比较简单。但大部分人基本是知其然,不知其所以然。譬如早期JDK只允许switch的表达式的值int及int类型以下的基本类型,后期的JDK却允许匹配比较字符串、枚举类型,这是怎么做到的呢?原理是什么?本文将深入去探索。
基础
我们现在使用的Java的版本,基本上是都支持String类型的。当然除了String类型,还有int、char、byte、short、enum等等也都是支持的。然而在其底部实现中,还是基于整型的,也就是int、byte、short这些类型。
我们先来看一下int的一个简单例子,主要部分源代码
publicstaticvoidmain(String[]args){ intn=2; switch(n){ case1: break; case2: break; case3: break; default: } }
再使用javac命令编译,javap命令反编译之后得到如下关键部分字节码:
0:iconst_2 1:istore_1 2:iload_1 3:tableswitch{//1to3 1:28 2:31 3:34 default:37 } 28:goto37 31:goto37 34:goto37 37:return
看不懂的话可以点击这里查看参考对照表。
当然懒得看的话我们也可以直接把class文件反编译成源代码,可以直接将class文件拖进IDEA,得到如下代码:
publicstaticvoidmain(String[]var0){ bytevar1=2; switch(var1){ case1: case2: case3: default: } }
这里总的来说和源代码变化不大,只是将int类型都转化为了byte类型。这里转化的原因,在于我们最初的case里面的值刚好在byte的范围之内。如果case的值稍微大点,它可能就会转化为short类型,再大点,就直接是int类型了。需要注意的是switch里面不支持float、long这些类型。
String类型讲解
有了上文的理解之后,下面应该会简单许多。
同样的,还是先上源代码
publicstaticvoidmain(String[]args){ Stringstr="sdf"; switch(str){ case"aaa": break; case"ccc": break; case"bbb": break; default: } }
然后编译之后丢进IDEA反编译得到反编译的代码
publicstaticvoidmain(String[]var0){ Stringvar1="sdf"; bytevar3=-1; switch(var1.hashCode()){ case96321: if(var1.equals("aaa")){ var3=0; } break; case97314: if(var1.equals("bbb")){ var3=2; } break; case98307: if(var1.equals("ccc")){ var3=1; } } switch(var3){ case1: case0: case2: default: } }
可以看到,String类型的switch,转换为了字符串的哈希比较,而其哈希返回的正是int类型。hash相同的情况再通过equals方法对比字符串的值,因此引进局部变量var3,是很有必要的。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对毛票票的支持。