2014年发布的java SE 8和2017年发布的java EE 8,至今还是使用最广泛的java版本,大部分java开发者对于java 8之后的升级总是敬而远之,这跟java 9以后的破坏性升级和oracle的商用协议有关,但随着9月java 17的发布,我们有更多理由去升级和在新项目中使用更新的java了。
人们在java 8这个版本不愿意升级,除了怕影响项目稳定性,还有就是java 9之后的发布频率太快,但在多了解一点java 9之后的更新策略后就会知道非LTS版本是不需要升级,也不建议升级的,只有每隔三年发布的LTS版本才有必要升级(另外java 17之后oracle有个提议改动LTS版本发布频率,之后的LTS版本可能每两年发布一次);另一方面是有很多破坏性更改,升级后旧项目可能直接报错。但在java 8发布至今的7年后,升级的解决方案已经很成熟了,这些问题不应该还是我们升级的阻力
关于升级时可能遇到的问题我做了个汇总,以版本区分,升级时,也建议一步步升级,比如先升级到java11,没问题再升级到java17,便于发现升级时的问题。以下解决方案基于maven
如果你的项目基于java 8,在升级前最好先升级依赖项,从java 8升级到java 17是一个很大的跨越,依赖项不升级则出问题的概率会比较高,maven可以用mvn versions:display-dependency-updates
命令检查依赖项更新,输出会类似这样
然后可以把依赖项升级到输出的对应版本,大部分包升级不会出问题,如果有问题,建议去出问题的依赖官方仓库寻找解决方案。这个命令是直接查询maven远程仓库,如果依赖项多的话会运行比较长的时间
java 11删除了这些原本在jdk中的包:
apt install fontconfig
javax.xml.bind.*
更改为jakarta.xml.bind.*
,下图列举了包名的改动,如果项目使用了这些包,需要在代码和pom.xml中更改相应包名删除了CMS GC,对于老项目或针对CMS专门调优过的项目,建议升级后使用G1 GC
Nashorn JavaScript Engine在这个版本被移除,如果使用了则需要手动添加这个依赖项
<dependency> <groupId>org.openjdk.nashorn</groupId> <artifactId>nashorn-core</artifactId> <version>15.2</version> </dependency>
java 16也是一个改动很大的版本,这个版本默认对jdk内部的很多api做了强封装,默认情况下不可访问(可以通过选项--illegal-access
更改这个行为,但官方不建议),这个主要影响一些工具,比如lombok,而lombok在java 16发布后不久更新了版本解决这个问题。
如果实在找不到兼容的方法,则在pom.xml修改compiler plugin参数可以解决问题:
<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compiler-plugin</artifactId> <version>3.8.1</version> <configuration> <fork>true</fork> <compilerArgs> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.comp=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.file=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.main=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.model=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.parser=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.processing=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.tree=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.util=ALL-UNNAMED</arg> <arg>-J--add-opens=jdk.compiler/com.sun.tools.javac.jvm=ALL-UNNAMED</arg> </compilerArgs> </configuration> </plugin>
如果升级到java 16,但lombok没有更新,则会报一个让你一头雾水的错误:
[ERROR] Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project broken: Compilation failure -> [Help 1]
除此之外没有更多信息了,这一点可以说是一个很大的坑
主要删除了一些实验性特性和老旧的API:
--illegal-access
参数不在可用,如果在java 17使用这个参数访问受限的api则会报出InaccessibleObjectException
,大多数情况下只要升级了依赖项是不会碰到这个情况的,但如果出现问题,则可以使用--add-opens
来对不可访问的api授权。对开发人员来说最想做的自然是使用新的特性,包括var records instanceof switch这些新关键字和旧语法的改进,以及Stream和Optional等API的改进等,此处不在赘述。
这里总结的是一些我自己升级过程中遇到的问题,只要将依赖项同步升级,基本可以解决升级会遇到的所有问题。当然不是所有项目都适合升级的,这里需要根据项目的情况仔细斟酌,如果是新项目,想跟上技术迭代的脚步,还是非常推荐升级到java 17的。
参考文章及链接: