JAVA字符串类型switch的底层原理详析
发布于 2020-4-24|
复制链接
摘记: 前言switch 语句是非常的基础的知识,掌握起来也不难掌握,语法比较简单。但大部分人基本是知其然,不知其所以然。譬如 早期JDK只允许switch的表达式的值 int及int类型以下的基本类型,后期的JDK却允许匹配比较 字符串、枚举类型,这是怎么做到的呢?原理是什么?本文将深入去探索。基础
我们现在使用的Java的版本,基本上是都支持String类型的。当然除了String类型,还有int、char、byte、short、e ..
前言switch 语句是非常的基础的知识,掌握起来也不难掌握,语法比较简单。但大部分人基本是知其然,不知其所以然。譬如 早期JDK只允许switch的表达式的值 int及int类型以下的基本类型,后期的JDK却允许匹配比较 字符串、枚举类型,这是怎么做到的呢?原理是什么?本文将深入去探索。基础我们现在使用的Java的版本,基本上是都支持String类型的。当然除了String类型,还有int、char、byte、short、enum等等也都是支持的。然而在其底部实现中,还是基于 整型的,也就是int、byte、short这些类型。我们先来看一下int的一个简单例子,主要部分源代码
```java
public static void main(String [] args){
int n = 2;
switch (n){
case 1:
break;
case 2:
break;
case 3:
break;
default:
}
}
```
再使用javac命令编译,javap命令反编译之后得到如下关键部分字节码:
```java
0: iconst_2
1: istore_1
2: iload_1
3: tableswitch { // 1 to 3
1: 28
2: 31
3: 34
default: 37
}
28: goto 37
31: goto 37
34: goto 37
37: return
```
看不懂的话可以 点击这里 查看参考对照表。
当然懒得看的话我们也可以直接把class文件反编译成源代码,可以直接将class文件拖进IDEA,得到如下代码:
```java
public static void main(String[] var0) {
byte var1 = 2;
switch(var1) {
case 1:
case 2:
case 3:
default:
}
}
```
这里总的来说和源代码变化不大,只是将int类型都转化为了byte类型。这里转化的原因,在于我们最初的case里面的值刚好在byte的范围之内。如果case的值稍微大点,它可能就会转化为short类型,再大点,就直接是int类型了。需要注意的是switch里面不支持float、long这些类型。String类型讲解
有了上文的理解之后,下面应该会简单许多。
同样的,还是先上源代码
```java
public static void main(String [] args){
String str = "sdf";
switch (str){
case "aaa":
break;
case "ccc":
break;
case "bbb":
break;
default:
}
}
```
然后编译之后丢进IDEA反编译得到反编译的代码
```java
public static void main(String[] var0) {
String var1 = "sdf";
byte var3 = -1;
switch(var1.hashCode()) {
case 96321:
if (var1.equals("aaa")) {
var3 = 0;
}
break;
case 97314:
if (var1.equals("bbb")) {
var3 = 2;
}
break;
case 98307:
if (var1.equals("ccc")) {
var3 = 1;
}
}
switch(var3) {
case 1:
case 0:
case 2:
default:
}
}
```