Java中的Switch都支持String了,为什么不支持long?

来源 | jitwxs.cn/6f3eddff.html

我们知道 Java Switch 支持byte、short、int 类型,在 JDK 1.5 时,支持了枚举类型,在 JDK 1.7 时,又支持了 String类型。那么它为什么就不能支持 long 类型呢,明明它跟 byte、short、int 一样都是数值型,它又是咋支持 String 类型的呢?

一、结论

不卖关子,先说结论:

switch 底层是使用 int 型 来进行判断的,即使是枚举、String类型,最终也是转变成 int 型。由于 long 型表示范围大于 int 型,因此不支持 long 类型。

下面详细介绍下各个类型是如何被转变成 int 类型的,使用的编译命令为 javac,反编译网站为:http://javare.cn

二、枚举类型是咋变成 int 类型的?

在没有实验之前,我想当然的认为它是不是根据枚举的 int 型字段来计算的(因为一般枚举都是一个int型,一个string型),但是转念一想,万一枚举没有 int 型字段呢,万一有多个 int 型字段呢,所以肯定不是这样的,下面看实验吧。

定义两个枚举类,一个枚举类有一个int型属性,一个string型属性,另外一个枚举类只有一个string属性:

public enum SexEnum {  MALE(1, "男"),  FEMALE(0, "女");  private int type;  private String name;  SexEnum(int type, String name) {  this.type = type;  this.name = name;  }  
}  
public enum Sex1Enum {  MALE("男"),  FEMALE("女");  private String name;  Sex1Enum(String name) {  this.name = name;  }  
}  

然后编写一个测试类,并且让两个枚举 switch 的 FEMALE 和 MALE 对应的返回值不同:

public class SwitchTest {  public int enumSwitch(SexEnum sex) {  switch (sex) {  case MALE:  return 1;  case FEMALE:  return 2;  default:  return 3;  }  }  public int enum1Switch(Sex1Enum sex) {  switch (sex) {  case FEMALE:  return 1;  case MALE:  return 2;  default:  return 3;  }  }  
}  

将这几个类反编译下:

// SexEnum.class  
public enum SexEnum {  MALE(1, "鐢�"),  FEMALE(0, "濂�");  private int type;  private String name;  // $FF: synthetic field  private static final SexEnum[] $VALUES = new SexEnum[]{MALE, FEMALE};  private SexEnum(int var3, String var4) {  this.type = var3;  this.name = var4;  }  }  // Sex1Enum.class  
public enum Sex1Enum {  MALE("鐢�"),  FEMALE("濂�");  private String name;  // $FF: synthetic field  private static final Sex1Enum[] $VALUES = new Sex1Enum[]{MALE, FEMALE};  private Sex1Enum(String var3) {  this.name = var3;  }  }  

反编译这两个枚举类,发现其中多了一个 $VALUES 数组,内部包含了所有的枚举值。继续反编译测试类:

// SwitchTest$1.class  
import com.example.express.test.Sex1Enum;  
import com.example.express.test.SexEnum;  // $FF: synthetic class  
class SwitchTest$1 {  // $FF: synthetic field  static final int[] $SwitchMap$com$example$express$test$SexEnum;  // $FF: synthetic field  static final int[] $SwitchMap$com$example$express$test$Sex1Enum = new int[Sex1Enum.values().length];  static {  try {  $SwitchMap$com$example$express$test$Sex1Enum[Sex1Enum.FEMALE.ordinal()] = 1;  } catch (NoSuchFieldError var4) {  ;  }  try {  $SwitchMap$com$example$express$test$Sex1Enum[Sex1Enum.MALE.ordinal()] = 2;  } catch (NoSuchFieldError var3) {  ;  }  $SwitchMap$com$example$express$test$SexEnum = new int[SexEnum.values().length];  try {  $SwitchMap$com$example$express$test$SexEnum[SexEnum.MALE.ordinal()] = 1;  } catch (NoSuchFieldError var2) {  ;  }  try {  $SwitchMap$com$example$express$test$SexEnum[SexEnum.FEMALE.ordinal()] = 2;  } catch (NoSuchFieldError var1) {  ;  }  }  
}  

首先生成了一个名为 SwitchTest$1.java 的链接类,里面定义了两个枚举数组,这两个数组元素添加的顺序完全和测试类中 switch 类调用的顺序一致。

图片

枚举元素在数组中的下标由 ordinal() 函数决定,该方法就是返回枚举元素在枚举类中的序号。

这里我们其实就已经知道了,在 switch 语句中,是根据枚举元素在枚举中的序号来转变成 int 型的。最后再看下测试类的反编译结果验证下:

// SwitchTest.class  
import com.example.express.test.Sex1Enum;  
import com.example.express.test.SexEnum;  
import com.example.express.test.SwitchTest.1;  public class SwitchTest {  public int enumSwitch(SexEnum var1) {  switch(1.$SwitchMap$com$example$express$test$SexEnum[var1.ordinal()]) {  case 1:  return 1;  case 2:  return 2;  default:  return 3;  }  }  public int enum1Switch(Sex1Enum var1) {  switch(1.$SwitchMap$com$example$express$test$Sex1Enum[var1.ordinal()]) {  case 1:  return 1;  case 2:  return 2;  default:  return 3;  }  }  
}  

三、String 类型是咋变成 int 类型的?

首先我们先知道 char 类型是如何变成 int 类型的,很简单,是 ASCII 码,例如存在 switch 语句:

public int charSwitch(char c) {  switch (c) {  case 'a':  return 1;  case 'b':  return 2;  default:  return Integer.MAX_VALUE;  }  
}  

反编译结果:

public int charSwitch(char var1) {  switch(var1) {  case 97:  return 1;  case 98:  return 2;  default:  return Integer.MAX_VALUE;  }  
}  

那么对于 String 来说,利用的就是 hashCode() 函数了,但是 两个不同的字符串 hashCode() 是有可能相等的,这时候就得靠 equals() 函数了,例如存在 switch 语句:

public int stringSwitch(String ss) {  switch (ss) {  case "ABCDEa123abc":  return 1;  case "ABCDFB123abc":  return 2;  case "helloWorld":  return 3;  default:  return Integer.MAX_VALUE;  }  
}  

其中字符串 ABCDEa123abc 和 ABCDFB123abc 的 hashCode 是相等的,反编译结果为:

public int stringSwitch(String var1) {  byte var3 = -1;  switch(var1.hashCode()) {  case -1554135584:  if(var1.equals("helloWorld")) {  var3 = 2;  }  break;  case 165374702:  if(var1.equals("ABCDFB123abc")) {  var3 = 1;  } else if(var1.equals("ABCDEa123abc")) {  var3 = 0;  }  }  switch(var3) {  case 0:  return 1;  case 1:  return 2;  case 2:  return 3;  default:  return Integer.MAX_VALUE;  }  
}  

可以看到它引入了局部变量 var3,对于 hashCode 相等情况通过 equals() 方法判断,最后再判断 var3 的值。

四、它们的包装类型支持吗?

这里以 Integer 类型为例,Character 和 Byte 同理,例如存在 switch 语句:

public int integerSwitch(Integer c) {  switch (c) {  case 1:  return 1;  case 2:  return 2;  }  return -1;  
}  

反编译结果为:

public int integerSwitch(Integer var1) {  switch(var1.intValue()) {  case 1:  return 1;  case 2:  return 2;  default:  return -1;  }  
}  

可以看到,是支持包装类型的,通过自动拆箱解决。

那万一包装类型是 NULL 咋办,首先我们知道 swtich 的 case 是不给加 null 的,编译都通不过,那如果传 null 呢?

答案是 NPE,毕竟实际还是包装类型的拆箱,自然就报空指针了。



往期推荐一文汇总 JDK 5 到 JDK 15 中的牛逼功能!
过滤器VS拦截器的4个区别,看完豁然开朗!
对象复制的7种方法,还是Spring的最好用!
关注我,每天陪你进步一点点!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/545015.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

什么是WebSocket,以及如何在Python中使用它?

什么是WebSocket? (What is WebSocket?) WebSocket is a communications protocol which provides a full-duplex communication channel over a single TCP connection. WebSocket protocol is standardized by the IETF as RFC 6455. WebSocket是一种通信协议&am…

QCon讲师对对碰——洪小军采访梁宇鹏:就是爱Golang

编者按:QCon北京2015将于4月23日~25日在北京国际会议中心召开。在大会开始之前,InfoQ推出了讲师对对碰栏目,邀请一些技术专家相互采访,碰撞出思维的火花。在QCon上,美图架构平台部门负责人洪小军将分享《美拍后端技术演…

网页之错误代码大全

winray整理~~~ 400 无法解析此请求。 401.1 未经授权:访问由于凭据无效被拒绝。 401.2 未经授权: 访问由于服务器配置倾向使用替代身份验证方法而被拒绝。 401.3 未经授权:访问由于 ACL 对所请求资源的设置被拒绝。 401.4 未经授权:W…

我们如何在Python中创建多行注释?

When we need to comment on multiple lines/statements, there are two ways to do this, either comment each line or create multiline comments (or block comment). 当我们需要对多个行/语句进行注释时,有两种方法可以执行此操作,或者注释每行或创…

final的8个小细节,听说只有高手才知道!你知道几个?

final关键字是一个常用的关键字,可以修饰变量、方法、类,用来表示它修饰的类、方法和变量不可改变,下面就聊一下使用 final 关键字的一些小细节。细节一、final 修饰类成员变量和实例成员变量的赋值时机对于类变量:声明变量的时候…

java实现的简单程序登录界面

2019独角兽企业重金招聘Python工程师标准>>> 这是我写的简单代码: 简单,没什么嚼头,作业贴,直接上代码。文件保存用户名和密码,输入密码错误3次退出程序。 [java] view plaincopy 01.public Login() throws…

js对浏览器内部及外部窗口宽度及高度使用测试

<!DOCTYPE html> <html><head><meta charset"utf-8" /></head><body><p id"demo"></p><script>//Part I 取浏览器内部窗口宽度及高度//该例显示浏览器窗口的高度和宽度&#xff1a;&#xff08;不包…

golang rsa密钥_如何在Golang的地图中检查密钥是否存在?

golang rsa密钥When you try to get the value of key in a map, you get two return values. The first value is the value of key and second is bool value which could be either true or false. If a key is present in the map then the second value will be true else…

try-catch-finally中的4个巨坑,老程序员也搞不定!

作者 | 王磊来源 | Java中文社群&#xff08;ID&#xff1a;javacn666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;在 Java 语言中 try-catch-finally 看似简单&#xff0c;一副人畜无害的样子&#xff0c;但想要真正的“掌控”它&#xff0…

CentOS7安装Python3.4 ,让Python2和3共存

为什么80%的码农都做不了架构师&#xff1f;>>> #CentOS7安装Python3.4 &#xff0c;让Python2和3共存 环境&#xff1a;CentOS7.1 需求&#xff1a;网络畅通 编译需要的一些包&#xff0c;酌情安装 yum groupinstall "Development tools" yum install z…

字节二面:优化 HTTPS 的手段,你知道几个?

由裸数据传输的 HTTP 协议转成加密数据传输的 HTTPS 协议&#xff0c;给应用数据套了个「保护伞」&#xff0c;提高安全性的同时也带来了性能消耗。因为 HTTPS 相比 HTTP 协议多一个 TLS 协议握手过程&#xff0c;目的是为了通过非对称加密握手协商或者交换出对称加密密钥&…

js使用location的方法实验

location.hostname 返回 web 主机的域名 127.0.0.1 location.port 返回 web 主机的端口 &#xff08;80 或 443&#xff09; 8020 location.protocol 返回所使用的 web 协议&#xff08;http:// 或 https://&#xff09; http: location.pathname 返回当前页面的路径和文件名…

求首位相连一维数组最大子数组的和

结对成员&#xff1a; 朱少辉&#xff1a;主要负责代码编写 侯涛亮&#xff1a;主要负责程序测试 题目:一个首尾相接的一维整型数组&#xff0c;其中有正有负&#xff0c;求它的最大子数组并返回它的位置。 思路&#xff1a;在求一维子数组的基础上&#xff0c;先输入一个含有N…

SpringBoot接口幂等性实现的4种方案!

作者 | 超级小豆丁来源 | www.mydlq.club/article/94目录什么是幂等性什么是接口幂等性为什么需要实现幂等性引入幂等性后对系统的影响Restful API 接口的幂等性如何实现幂等性方案一&#xff1a;数据库唯一主键方案二&#xff1a;数据库乐观锁方案三&#xff1a;防重 Token 令…

scala 数组合并_Scala程序合并两个数组或数组缓冲区

scala 数组合并Scala | 合并两个数组 (Scala | Merging two arrays) Arrays are important data structures in programming and there may arise times when we have two different arrays and we need to merge them into one for processing. This is the case when you nee…

js的navigator对象的使用(浏览器信息)

window.navigator 对象包含有关访问者浏览器的信息。 <!DOCTYPE html> <html> <body> <div id"example"></div><script>txt "<p>Browser CodeName: " navigator.appCodeName "</p>"; txt &…

http中指定使用worker/prefork

注意&#xff1a; 2.4之前版本默认为prefork&#xff0c; 2.4已经变为event模式。三种模式比较&#xff1a; http://www.cnblogs.com/fnng/archive/2012/11/20/2779977.html在编译apache的时候&#xff0c;有一个参数叫做 --with-mpm... 等号后边用于指定那种模式&#xff…

Redis为什么变慢了?一文详解Redis性能问题 | 万字长文

Redis 作为优秀的内存数据库&#xff0c;其拥有非常高的性能&#xff0c;单个实例的 OPS 能够达到 10W 左右。但也正因此如此&#xff0c;当我们在使用 Redis 时&#xff0c;如果发现操作延迟变大的情况&#xff0c;就会与我们的预期不符。你也许或多或少地&#xff0c;也遇到过…

Java PipedOutputStream flush()方法与示例

PipedOutputStream类flush()方法 (PipedOutputStream Class flush() method) flush() method is available in java.io package. flush()方法在java.io包中可用。 flush() method is used to flush this PipedOutputStream and forces bytes to be written out of any buffered…

蜕变成蝶~Linux设备驱动之字符设备驱动

一、linux系统将设备分为3类&#xff1a;字符设备、块设备、网络设备。使用驱动程序&#xff1a; 字符设备&#xff1a;是指只能一个字节一个字节读写的设备&#xff0c;不能随机读取设备内存中的某一数据&#xff0c;读取数据需要按照先后数据。字符设备是面向流的设备&#x…