从Java8到Java17bb

背景

Java8的前世今生

作为一名程序员,笔者从大学开始接触Java这门编程语言,还记得当时的版本是1.4,不过这个版本的寿命还挺长,目前在一些金融行业的存量系统中依然还有1.4版本的影子。从C/C++到Java,Java给我的第一印象,就是这门语言的语法非常之啰嗦,比如内部类,像下面这段排序的代码,在C语言里面可以很方便地通过函数指针来解决,C++可以通过运算符重载来解决,唯独Java语言的写法最为冗长,这可能跟在Java的世界里一切皆是对象的这一设计理念相关。
大学毕业参加工作后,开始用到JDK1.6,带来的最大感受是终于支持泛型了,操作集合的时候终于不用各种类型转换了,即便他的实现还是个假的泛型,但是带来的方便也是非常明显的。转眼到了2014年3月份,今天的第一个主角Java8正式GA,这是一个里程碑式的版本,带来了巨量的升级,其中包括对函数式编程的支持,像上述的冗长的排序代码,通过使用一行代码即可搞定;除此之外,还包括对Streams支持,方便了开发者写出更加简洁、高效的代码;然后,在Java8中还引入了全新的日期操作API,极大地方便了开发人员对日期和时间的操作,关键是他还是线程安全的,原来那套旧的日期和时间操作的API极度难用,而且容易出错。最后,还引入了Optional这一特性,极大地方便了对NPE问题的处理。这些新的特性的加入,为Java这门语言注入了新的生机,大量的开发人员从中受益,从这也可以看出Oracle官方对这门语言还是非常地重视,也投入了大量的资源持续进行优化升级,所以大家可以继续依赖这门语言。

Oracle的商业支持

图1是2020年统计的Java各版本的使用情况,Java8的占比还是非常之大的,足以说明这个版本取得了非常大的成功,也不得不承认Java8是一个划时代的版本,从另一方面也说明了Java8之前的版本确实也太差了。竞争就是这样,不进步就被淘汰。从Java8之后,Java的更新也开始进入了一个快车道,每半年就会发布一次数字更新,但很多都是临时版本,长期维护的版本则由Java自行宣布,当前8,11,17属于是LTS(长期维护版本),表1中列出了各个版本的官方维护时间。
notion image
图 1 2020年编程语言使用排名
Release
GA Date
Premier Support Until
Extended Support Until
Sustaining Support
7 (LTS)
July 2011
July 2019
July 2022
Indefinite
8 (LTS)
March 2014
March 2022
December 2030
Indefinite
9 (nonLTS)
September 2017
March 2018
Not Available
Indefinite
10 (nonLTS)
March 2018
September 2018
Not Available
Indefinite
11 (LTS)
September 2018
September 2023
September 2026
Indefinite
12 (nonLTS)
March 2019
September 2019
Not Available
Indefinite
13 (nonLTS)
September 2019
March 2020
Not Available
Indefinite
14 (nonLTS)
March 2020
September 2020
Not Available
Indefinite
15 (nonLTS)
September 2020
March 2021
Not Available
Indefinite
16 (non-LTS)
March 2021
September 2021
Not Available
Indefinite
17 (LTS)
September 2021
September 2026
September 2029
Indefinite
18 (non-LTS)
March 2022
September 2022
Not Available
Indefinite
19 (non-LTS)
September 2022
March 2023
Not Available
Indefinite
20 (non-LTS)
March 2023
September 2023
Not Available
Indefinite
21 (LTS)
September 2023
**September 2028 **
September 2031
Indefinite
表 1 Oracle对Java版本支持路线图
从表1中我们可以得出一些有用的信息,当然了,如果是付费企业可以忽略,付费企业是上帝。
  1. Java8将在2022年3月份停止维护,如果这之后遇到问题,有两种办法,第一,自己想办法解决,这对一些技术比较强的IT企业可能并不是什么难题,但是技术一般的企就不一定能搞得定了。第二,花钱买服务,找官方支持。相信国内99%以上的企业用的都是在免费使用Java。
  1. Java11也并不是一个安全和稳妥的升级目标版本,在2023年7月份,Oracle官方对Java11的支持也将结束,此时,替从Java8迁移到Java11的用户感到有些不值,下次升级一定要看准了再升,当然了,如果是新应用直接使用Java11,那么后面基本可以无感升级到后续版本。
  1. Java8将在2022年3月停止维护,那么下一次值得升级和使用的版本就是Java17了,这个版本在2021年11月正式GA,官方将公开支持到2026年11月份,而且也是一个LTS版本,更为关键的是,这个版本可以免费商用,此时此刻全场欢呼有没有。

持续充电

作为一名程序员,保持持续充电是非常有必要,在Java8即将停止得到官方免费升级支持的时候,相信会有越来越多的项目将基于Java17进行构建和迁移,包括于2022年1月20号,大名鼎鼎的SpringBoot3.0释放第一个里程碑版本M1,该项目就是基于Java17构建的,是的,你没看错,这个项目就是使用Java17进行构建,依赖的相关组件很快将全部会进行升级,届时如果想使用一些新的特性,在Java8下将无法使用,因为根本无法通过编译,相信这将会是一个痛苦的开始。因此,在此时此刻,突然有一种冲动,突然想在这一块给自己充充电,停留在Java8的时代确实也有些年头了,担心再不学习和充电,后面新的开源代码还有项目中的新的源代码将看不懂了,届时可能被这个时代给抛弃了。IT就是这样,知识的更迭速度非常非常快,持续充电非常必要。

收益

升级和使用新版本的Java带来的收益还是比较可观的,这里以升级Java17为例,首先可以借助新版本中新的语言特性和新的Java标准SDK,编写出更易于阅读和维护的代码,原来需要10行代码才能实现的功能,现在或许只需要一行代码就可以搞定了。在性能上,Java17还带来了不小的提升,具体可以参考相关测试链接,也就是说仅仅通过升级Java版本,就可以带来一次性能上的提升,想想还是挺划算的。新版本中还增加了一些新的库函数,比如更加好用的Http客户端API,终于可以摆脱Apache的HttpClient这个开源组件了。还有Record记录类这一新特性的加入,给经常写PO、DO、VO的开发人员也带来了福音。其他特性这里不再赘述,后面主要篇幅将详细的介绍从Java8到Java17这中间的一些变化和新的特性,通过学习和了解这些内容,能很快地从Java8转到Java17,当然了,对于有些新的特性,还是亲自动手去玩一遍,可能印象会更加深刻,效果也会更加好一些。

特性介绍

从Java8到Java17这中间增加的特性非常多,有些特性增加完后很快又被废弃,有些经过好几个版本的更迭后才被正式启用。图2是笔者总结归纳的从Java8到Java17增加的一些重要的新特性,有些特性可能跟我们关系不大,或者根本接触不到,或者一些几乎没用到过的一些的删除,这些点没有在这里列举。这里仅仅列举了对我们开发和使用关系非常密切的,能方便开发者且有明显感知得到的特性,如果对其他特性感兴趣,请访问Java官方链接。下面按照图2中从上到下的顺序详细进行介绍。
notion image
图 2 Java8到Java17重要新特性汇总

语法新特性

文本块

这个更新非常有用的,在没有这个特性之前,编写一段长文本是非常痛苦的,虽然有像IDEA这样的集成编辑工具自动处理,但是最后呈现出来的效果也是很丑陋的,里面一堆拼接符号。现在在这个字符串块内随便写,包括但不限于JSON、HTML和SQL等等看起来也非常清爽,给这个新特性五颗星,以后只需要关注字符串本身了,而不需要开发者关心拼接操作了。

NullPointerException增强

这一功能非常的强大而又实用,相信每一位Javaer等这个功能实在等得太久了。每一位Java程序员一定都查过NPE问题,也就是大名鼎鼎的空指针问题,而且对这类问题一定也非常痛恨,为什么呢?因为报的错误信息里面没有反馈是哪个对象的问题,就只会摆抛出来一个NullPointerException和一堆用处不大的堆栈信息,然后勤劳的程序员们就开始去定位问题去了,一个小时,两个小时过去了,问题可能还未定位出来。尤其是在遇到喜欢炫技开发者们,在一行代码上级联调用,鬼知道哪个对象为空导致报错了,只能从前往后推。如果是在测试环境,很多同学可能会想到使用远程debug来查明是哪个对象为空了。但为了查明是哪个对象是为空对象都需要这样大费周章,这到底是哪个地方的问题?为了规避出现这样的问题,阿里的编码规范中甚至出现了不允许级联调用,但这也只是治标不治本,在JDK17中,这个问题总算得到彻底解决了。如下代码所示,两个测试用例都会抛出NPE异常。
运行后得到的报错如下,报错信息非常清晰地指出来了是哪个对象为空导致出现的NPE问题,无论级联调用得有多深,都能很清晰地指示出来,非常好用,非常强大,给这个功能五颗星,程序员以后终于可以早些下班了。
java.lang.NullPointerException: Cannot invoke "java.lang.Integer.longValue()" because the return value of "java.util.ArrayList.get(int)" is null
at NullPointerExceptionEnhancer.test1(NullPointerExceptionEnhancer.java:20)
at NullPointerExceptionEnhancer.main(NullPointerExceptionEnhancer.java:12)
Exception in thread "main" java.lang.NullPointerException: Cannot invoke "Object.equals(Object)" because "obj" is null
at NullPointerExceptionEnhancer.test2(NullPointerExceptionEnhancer.java:28)
at NullPointerExceptionEnhancer.main(NullPointerExceptionEnhancer.java:13)

Records

在Java中,POJO是被大家所熟知的,包括DO、PO、VO、DTO等,都是普通的POJO对象,一个POJO对象包括一系列的类成员变量和相应的Getter和Setter函数,清一色的模板代码,虽然可以通过工具或者IDE直接生成,但是修改维护起来还是非常麻烦的,为了解决这个问题甚至还出现了像lombok这样的插件,在编译期间自动生成相应的Getter和Setter,甚至像hashcode、equals、构造函数、builder构造器模式模板代码都给自动生成了,使用起来也算是相当方便。但是这毕竟也只是个插件,而且一旦使用这个插件,那么项目团队里面所有人都得使用,IDE上面还都得安装相应的插件,有一种强迫的意思,实属不太友好。就在这个时候,Java来了一个标准解决方案,即Records这一语言特性,示例代码如下所示。record这一特性的引入,非常干净利落地解决了上述部分问题。但是如果要实现相应的hashcode和equals方法,那么还是得手工去写了,但好在有IDE自动生成的功能。

全新的switch表达式和模式匹配

有人可能问了,Java语言不早已支持switch了嘛,有什么好提的?讲真,这次的提升还有必要好好地来聊一聊了。在JDK12的时候就引入了switch表达式,注意这里是表达式,而不是语句,原来的switch是语句。如果不清楚两者的区别的话,最好先去了解一下。主要的差别就是就是表达式有返回值,而语句则没有。再配合模式匹配,以及yield和“->”符号的加入,全新的switch用起来爽到飞起来,具体请看对比示例。笔者在玩了几把新式的写法后,最大的感受就是简洁明了,而且还不易出错,如果习惯了新式的写法,那么再也不用担心少写break语句而导致出现bug了,相信绝大多数Java程序员都犯过漏写break语句而导致出现一些稀奇古怪的问题吧,新技能get起来。此外,模式匹配的引入,也确实给程序员带来了很大的便利。最后,case后面居然支持null值了,小小的惊喜。

私有接口方法

从Java8开始,允许在interface里面添加默认方法,其实当时就有些小困惑,如果一个default方法体很大怎么办,拆到另外的类去写吗?实在有些不太合理,所以在Java17里面,如果一个default方法体很大,那么可以通过新增接口私有方法来进行一个合理的拆分了,为这个小改进点个赞。

局部变量类型推断

在Java8的时候,其实就已经支持了类型推断了,在对Stream进行操作的时候,比如下面这段代码,在执行filter的时候里面的lambda表达式中的参数都没有声明类型,这里其实就用到了类型推断。
int maxWeight = blocks.stream().filter(b -> b.getColor() == BLUE).mapToInt(Block::getWeight) .max();
在Java10的时候又引入了另外一个特性,叫局部变量类型推断这个特性。正如这个特性的名称,这个特性只能用于局部变量,且类型是确定的,无二义性的,下面的示例代码中给出了哪些地方不能用局部变量类型推断,也就是var关键词在哪些场景下不允许使用。

密封类

通过sealed修饰符来描述某个类为密封类,同时使用permits关键字来制定可以继承或实现该类的类型有哪些。注意sealed可以修饰的是类(class)或者接口(interface),所以permits关键字的位置应该在extends或者implements之后。这个特性跟final是有区别的,final修饰的类表示该类不允许被继承,不允许修饰interface和抽象类,此外,被密封的类必须指定能被继承的子类,且子类也必须被标为sealed或者non-sealed或者final。这个特性可以防止类被误继承,把继承的范围固定在某个固定可控的范围类。
以下为描述一个接口为密封类的写法实例:

JavaSDK更新

全新的HttpClient

这个API首次出现在9之中,不过当时并非是一个稳定版本,在Java11中正式得到发布,所以在Java17里面可以放心地进行使用。原来的JDK自带的Http客户端真的非常难用,这也就给了很多像okhttp、restTemplate、Apache的HttpClient和feign这样的第三方库极大的发挥空间,几乎就没有人愿意去用原生的http客户端的。但现在不一样了,感觉像是新时代的API了。FluentAPI风格,处处充满了现代风格,用起来也非常地方便,再也不用去依赖第三方的包了,就两个字,清爽。

集合类的工厂方法

在Java8的年代,即便创建一个很小的集合,或者固定元素的集合,那么也必须经历如下这几步。新建一个集合类,然后调用相应的函数增添元素,而且该集合还是一个可变的集合。此外,像List、Set和Map接口都增加了一个copyOf的default方法,用于做集合元素的深拷贝,这个方法也非常的实用。
);
这,其实很不像一个现代化语言。所以现在可以用简单的写法:
包括List,Set,Map都可以使用of来直接创建集合。

Flow API

这个东西的前身叫rxjava,或者叫reactive java,后来又出现了一个叫project reactor。事件驱动已经不能简单的用来概括响应式编程了,为什么要增加这样的API呢,因为那段时间各个大厂纷纷发力异步编程,弄出很多种所谓的响应式框架,各自都宣称自己的性能好。后来被某个开源联盟联合起来一起出了一个响应式的规范,主要包括下面四大块。
  • Subscriber:订阅者
  • Publisher:生产者
  • Subscription:订阅关系
  • Processor:订阅者和生产者之间的N个处理步骤
响应式编程并不能提升多少性能,而是使程序更加稳定和获得更好的扩展性。

Stream API的增强

增加takeWhile, dropWhile, ofNullable, iterate以及toList的API,越来越像一些函数式语言了。用法举例如下。

新的String方法

  • repeat:重复生成字符串
  • isBlank:不用在引入第三方库就可以实现字符串判空了
  • strip:去除字符串两边的空格,支持全角和半角,之前的trim只支持半角
  • lines:能根据一段字符串中的终止符提取出行为单位的流
  • indent:给字符串做缩进,接受一个int型的输入
  • transform:接受一个转换函数,实现字符串的转换

InputStream和Reader之transferTo方法

在没有transferTo方法前,若要将一个I/O输入转成一个I/O输出,那么会怎么做呢?首先,从I/O输入中读一段数据到某个buffer中,然后,将这个buffer中的数据写到I/O输出中,循环此动作,直到数据完全读完并写入到I/O输出,这中间还有考虑很多边界的问题,完成这个动作还是比较繁琐的。在有了transferTo之后,可以通过一个简单的方法调用,就可以把这一堆琐事全部搞定了,如下代码所示,是不是非常方便了。

新工具

jshell

在新的JDK版本中,支持直接在命令行下执行java程序,类似于python的交互式REPL。简而言之,使用 JShell,你可以输入代码片段并马上看到运行结果,然后就可以根据需要作出调整,这样在验证一些简单的代码的时候,就可以通过jshell得到快速地验证,非常方便。

java命令直接执行java文件

在现在可以直接通过执行“java xxx.java”,即可运行该java文件,无须先执行javac,然后再执行java,是不是又简单了一步。

其他

模块化

大家都知道Java的package,需要的时候引入(import)一下对应的类,module在package的上面一层,一个模块都会有一个module-info.java的描述文件,来描述当前模块需要引入哪些package和对哪些package可见。
这么做的好处,首先对于jdk本身来说我不需要的package我就不要了引入了,不像现在会默认塞一个rt.jar进来,这个东西有60M多,但大部分的内容都没用到。然后没有在module-info.java中声明的exports,即使是public的类在外部也无法被使用,这就进一步增强了安全性。
jdk身先士卒首先将rt.java拆成了很多个小的jmod文件,但好像也仅限于此,热门框架跟进的不积极。基于现有的Spring boot+maven的管理模式,大到依赖管理小到bean的管理都井井有条。如果各大框架进一步拆解自己的变成一个个模块,那可能确实打出来的jar会小一些,但可惜多数是没有响应的。

模块和maven不是一个东西

这个依赖管理让很多人一眼就想到了maven,但两者解决的不是一个问题。maven只能管到jar包的依赖,而module会深入到类中去做更精细的依赖管理和权限控制。而maven本身还有更重要的活去干:编译、测试、打包、发布等。
鉴于确实没多少人用,我也很难下结论说他好还是不好,只是现有的项目想迁移成本是极高的,新的项目不用module也可以实现大部分的功能,只是jar会比较大一些,不过在容器本身的体积面前确实也不算什么。

ZGC

在ParallelOldGC、CMS、G1之后,JDK11带来的全新的ZGC,全称为The Z Garbage Collector,听这名字取得就挺牛逼的。官方声称垃圾回收的停顿时间不超过10ms且最大支持16TB的堆,并且停顿时间不会随着堆的增大而增加。那到底为我们解决了什么问题呢?Oracle官方介绍它是一个Scalable、Low Latency的垃圾回收器。所以它的目的是降低停顿时间,由此会导致吞吐量有所降低。但是吞吐量降低问题不大,横向扩展几台服务器就能解决问题。对于ZGC,目前官方已经建议可以用于实际生产应用中,这一定是未来一个通用的垃圾回收器,要想了解更多,请参考官方链接

改进的JavaDoc

新版本Java中,生成成的Java文档已经支持HTML5,且支持全文搜索功能。

总结

Java8的生命的尾声或许即将来临,或许在存量的系统中继续发光发亮,但无论是哪一种,Java17的使用量将会逐年攀升,越来越多的开源项目也将会升级到Java17,SpringBoot3.0只是一个开始,你准备好了吗?
Loading...
目录
文章列表
王小扬博客
产品
Think
Git
软件开发
计算机网络
CI
DB
设计
缓存
Docker
Node
操作系统
Java
大前端
Nestjs
其他
PHP