史上将switch原理讲得最透彻的文章
yuyutoo 2024-12-22 21:45 1 浏览 0 评论
大家好,我是贠学文,点击右上方“关注”,每天为您分享java程序员需要掌握的知识点干货。
上一期说了一下java判断语句中if和switch在使用上的区别,以及个人的使用习惯,今天来说一下他们在底层的实现原理。当然了,if的实现原理非常简单,就是顺序查找,没啥可说的,但是switch的原理,还是值得说一下的。
一、switch支持的类型
1.byte,short,char,int四种基本类型
2.byte,short,char,int四种基本类型对应包装类
3.String类型
4.enum类型
其实从严格意义上来讲,switch的底层,只支持int一种类型,但是其他类型可以通过各种方式转换为int类型,每种类型的转换方式如下:
1.byte,short,char三种基本类型可以隐式地自动转换为int类型
2.byte,short,char,int四种基本类型对于的包装类,实际上是做了自动拆箱,先转换成各自对应的基本类型后,在隐式的转换为int类型
3.String类型,取的是String对象的hashCode值,hashCode就是int类型
4.enum类型,enum类型实际上就是一个数组,取得是数组的下标。
但是看到这里大家会不会有一个疑问,那就是每种整数类型的都支持了,为什么没有支持long类型呢?这个问题我们后面会解答。
二、switch的实现原理
我们看下图1所示的代码以及图2对应的字节码:
我们在图1的代码中,case的分支条件有从11到20,这10个分支条件,那图2字节码中我标红的部分,是一个tableswitch指令,我们可以把这个指令的内容理解成一个数组,数组中的每个元素,冒号前面的,是记录的我们case分支条件的值,即11到20,此时我们记录start=11,end=20,而冒号后面的,即定位到这个分支条件后,要跳转到的执行代码的行数。如果你想查找某个值key,它会首先看你这个值是否在start和end的范围内,如果在,则直接用key-start,计算出数组的下标,找到数组中的元素,如果不在,则直接执行default部分,这种的查找方式,时间复杂度为O(1),效率非常高。
但是这是有人可能会问了,你图一的代码中,case分支的条件的值都是连续的,那如果不是连续的呢?是不是就不能用这种算法了。我们可以看下图3的代码:
我们删掉了几个分支条件,让其不是连续的,然后我们在看图4的字节码:
发现其实它会自动的把缺少的分支条件,自动的补齐,使其成为连续的,此时这种算法还是生效的。
但是我们图3的代码,虽然它的分支条件不连续,但是每个分支条件的值还都比较集中,这样它自动补齐的时候,需要补齐的数据也非常少,但是如果分支的条件不集中呢?这时如果再自动补齐,就需要补齐大量的数据,严重的浪费空间,所以这时候,这种算法就不合适了,我们可以看下图五的代码以及图六对应得字节码:
图5代码中,case分支条件非常不集中,所以我们在看图6的字节码中,这时就不在使用tableswitch指令了,而是使用了lookupswitch指令,这个指令了内容中,没有像tableswitch指令一样实现自动补齐,所以就不能根据数组下标去查找了,这时采用的二分查找的方式。
三、switch为什么不支持long?
1.上面我们在讲到switch原理的时候,提到了两个指令,一个tableswitch,一个lookupswitch,这两个指令中,支持的最大的范围就是int,所以无法支持long。
2.int的范围已经足够大了,已经到了20多个亿了,我们在日常的代码编写中,很少会有对这么大的数据做switch的,我觉得java的设计者,也是考虑到这一点,觉得没有必要支持long类型。
四、switch方式如何支持String类型
我们可以看下图7所示的代码和图8对于的字节码:
我们发现其实他是在底层调用了String的hasecode方法,然后把返回值作为case的分支条件,从而支持String类型。
五、switch方式如何支持enum类型
我们可以看下图九所示的代码和图十对于的字节码:
我们发现其实他是在底层调用了enum的ordinal方法,然后把返回值作为case的分支条件,从而支持enum类型。
到这里呢,switch的原理就说完了,但是还是像昨天说的一样,我在日常的编码中,几乎从来不用switch,虽然说它的效率要比if高,但是这种效率的差异,只有在分支条件特别多的时候才能体现出来。但是如果分支条件特别多了,不管用if还是switch,对后期的维护都是一个灾难。所以我在分支条件少的时候,会选择用if,当分支条件多的时候,我会用合适的设计模式来处理。
往期精彩:
作者介绍:
贠学文,具有多有经验的java开发工程师,业余时间利用头条分享技术知识点与自己对技术的感悟,帮助对自己未来感到迷茫的程序员,在技术上得到提升。结识一些志同道合的朋友,相互促进,共同进步。
相关推荐
- GitHub精选 | 基于go开发的定时任务管理系统
-
《GitHub精选》是我们分享Github中优质项目的栏目,包括技术、学习、实用与各种有趣的内容。本期推荐的是gocron-定时任务管理系统之前有过一期...
- linux定时任务,让你更深入的了解系统
-
cron的软件包cronierpm-qlcronierpm–qlcrontabs最关键的是一个叫做crontab的命令,我们要书写的非系统定时任务就是通过此程序来编写的;要保证crond.se...
- linux定时任务详解 linux定时任务执行
-
配置方式:(记忆口诀:分、时、日、月、星)#Fordetailsseeman4crontabs#Exampleofjobdefinition:#.---------------...
- 我终于会写 Java 的定时任务了 java定时任务指定时间执行
-
前言学过定时任务,但是我忘了,忘得一干二净,害怕,一直听别人说:...
- 如何用Windows计划任务设置:定时关机
-
定时关机其实是一个小功能,很多时候还真的需要它。你在网上找还真不是那么好找,下面我们用Windows计划任务设置:定时关机,而且,我还在计划任务中看到了几个不该有的计划任务。?打开:计划任务点击Co...
- 定时任务优化总结(从半个小时优化到秒级)
-
整体优化思路:1.按需查询、2.分小批次游标查询、3.JED场景下按数据库分片分组更新、4.精准定位要处理的数据、5.负载均衡业务背景:...
- 信创终端操作系统上定时任务crontab详解 | 统信 | 麒麟 | 中科方德
-
原文链接:信创终端操作系统上定时任务crontab详解|统信|麒麟|中科方德...
- VIVO手机定时任务功能_定时开关机、定时振动模式
-
手机在生活中是必不可少的,有些功能我们想要他定时切换变更,本文就来分享一下Vivo手机的定时任务功能。Vivo手机如何进入定时任务界面...
- 下班总是忘记关电脑!添加任务自动定时关机,不用再跑一趟!
-
我们每个人都有过这样的经历,下班后、或者出差,单位里的电脑总是忘记关机!很是烦恼!经常忘关电脑的话,一是费电,二是不安全,会给网络攻击留有足够的时间,会导致信息泄露,存在安全隐患。其实只要我们在电脑...
- 为什么定时任务到时间不执行?带你深入源码找答案
-
前言早上研发经理给我分配了一个开发任务:每5秒统计一次APP在线人数,并问我啥时能上线?我心想这需求用Spring的定时任务完美解决啊!作为一个萌新正好借此机会在经理面前表现一番,于是我拍着胸脯跟经理...
- 我来总结下 几种定时任务的执行方式
-
首先带入我们的业务场景:我们买火车票或者叫外卖的时候,下完单之后会跳转到支付页面,页面会有一个计时器,要求在指定时间内完成支付,否则订单自动取消。这是延时任务的一个典型场景,分析这个场景,就是如何在订...
- 聊聊定时任务的六种模式 定时任务详解
-
这篇文章,我们聊聊实现定时任务的六种策略。1自定义单线程上图中,我们启动一个线程,该线程无限循环执行,每隔20毫秒执行业务代码。...
- windows如何实现定时任务?配合脚本使用结局很满意
-
序言:作为一名程序员,通过定时任务去实现各种所需的功能是必须要掌握的,本文来讲解一下在windows服务器上如何创建定时任务。有需要的小伙伴赶紧收藏转发吧。第一步:打开控制面板-》系统和安全-》管理工...
- 电脑(计算机)如何定时执行任务 电脑定时器怎么定时
-
电脑想定时执行程序,任务计划步骤如下:1,系统服务里,确保"TaskScheduler"服务启动。2,在控制面板->系统和安全里,找到任务计划,并点击: 或在“管理工具”里点...
- 分布式定时任务最全详解(图文全面总结)
-
分布式定时任务是非常核心的分布式系统,下面我就全面来详解分布式定时任务以及分布式定时任务框架@mikechen本篇已收于mikechen原创超30万字《阿里架构师进阶专题合集》里面。什么是分布式定时任...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)