为什么switch里的case没有break不行

 前言

一个小姐姐拿着一个switch的选择题来问我。

成都创新互联公司,为您提供成都网站建设重庆网站制作、网站营销推广、网站开发设计,对服务汽车玻璃修复等多个行业拥有丰富的网站建设及推广经验。成都创新互联公司网站建设公司成立于2013年,提供专业网站制作报价服务,我们深知市场的竞争激烈,认真对待每位客户,为客户提供赏心悦目的作品。 与客户共同发展进步,是我们永远的责任!

之所以这么笃定地回答这个问题,并不是我知道其中原理,而是之前在一个群里,有人问了同类型的问题,我瞥了一眼记住了答案,所以才依葫芦画瓢。

小姐姐接着问我为什么,我说少个break,但凡再问一句:为什么少个break结果就不一样,我就回答不出来了。所以,为了将尴尬扼杀于摇篮,还是研究一下break在switch的作用。

 从字节码出发

按照惯例,先写demo表述问题。 

 
 
 
 
  1. public static void main(String[] args) {  
  2.    int i = 0;  
  3.    switch (i) {  
  4.        case 0:  
  5.            System.out.println(0);  
  6.        case 1:  
  7.            System.out.println(1);  
  8.        case 2:  
  9.            System.out.println(2);  
  10.  } 

运行代码,结果如下:

*明明只匹配了case 0,为什么1和2也执行了? 很费解!按照惯用套路,看看字节码能不能给个答案。

javac编译和javap查看:

「tableswitch」和「lookupswitch」都用于switch条件跳转,前者用于case值连续,例如上面代码中的0、1、2;后者用于case值不连续。

从字节码可以看出:switch中的case条件和对应代码块是分开的。如上图,case为0时,跳转到标号28代码处;为1时跳转到标号35代码处;为2时跳转到标号43代码处;default则跳转到标号49代码处。

这不,答案就出来了,当case 0匹配了之后,直接跳转到标号28代码处开始执行,输出0,然后策马奔腾,一路小下坡,顺序执行完后面所有代码,直到标号49 return,方法完执行完成,程序结束。

如果按照正常的思维,是不是case 0匹配之后,跳到28,执行完28、31、32输出0之后,就应该直接跳走,直接执行49。那么,这个"跳走”用字节码应该怎么表示?

用return?那不行,因为return会结束方法,这样switch后代码也无法执行。那怎么办嘞.... 

关于goto

goto:无条件跳转,goto 1表示跳转到标号1的代码处。

再写代码样例,这次在代码中给每个case都加上break。 

 
 
 
 
  1. public static void main(String[] args) {  
  2.     int i = 0;  
  3.     switch (i) {  
  4.         case 0:  
  5.             System.out.println(0);  
  6.             break;  
  7.         case 10:  
  8.             System.out.println(1);  
  9.             break;  
  10.         case 2:  
  11.             System.out.println(2);  
  12.             break;  
  13.     }  
  14.     System.out.println("Hello World");  

重新编译,再来看看字节码。

如图,与第一次的字节码相比,在标号35、45都有了goto指令。如果case 0匹配成功,则跳到标号28执行,执行完代码块对应的31、32指令之后,执行35的goto指令跳转到标号55,这样就跳出了switch作用范围,case 1和2也不会被执行。

等等,怎么少了一个goto,在标号55的上方应该还有一个goto才对!其实这就涉及到了编译器优化技术,最后一个goto也是跳转到标号55的指令,但没有goto下一步也一样顺序执行此行指令,所以这个goto被编译器视为无用代码进行了消除。 

switch和if区别

先用if实现上面switch逻辑。 

 
 
 
 
  1. public static void main(String[] args) {  
  2.     int i = 0;  
  3.     if (i == 0) {  
  4.         System.out.println(0);  
  5.     } else if (i == 1) {  
  6.         System.out.println(1);  
  7.     } else if (i == 2) {  
  8.         System.out.println(2);  
  9.     }  

编译成字节码:

「if_icmpne」用于比较两个int数。从字节码也可以看出if和switch的区别:if条件和代码块的字节码是顺序的,switch条件和代码块是分开的;if自动生成goto指令,switch只有加了break才生成goto指令。 

结语

case中的break告诉前端编译器:「给每个case对应代码块的最后加上goto」。这样,执行完匹配上的代码之后,就可以略过后面的case代码块了。

果然,求(xiao)知(jie)欲(jie)才是学习新知识的动力。

网站题目:为什么switch里的case没有break不行
URL地址:http://www.shufengxianlan.com/qtweb/news31/507381.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联