java9遷移注意問(wèn)題總結(jié)
序
本文主要研究下遷移到j(luò)ava9的一些注意事項(xiàng)。
遷移種類(lèi)
1、代碼不模塊化,先遷移到j(luò)dk9上,好利用jdk9的api
2、代碼同時(shí)也模塊化遷移
幾點(diǎn)注意事項(xiàng)
不可讀類(lèi)
比如sun.security.x509,在java9中歸到j(luò)ava.base模塊中,但是該模塊沒(méi)有export該package
可以通過(guò)運(yùn)行的時(shí)候添加--add-exports java.base/sun.security.x509=ALL-UNNAMED來(lái)修改exports設(shè)定
內(nèi)部類(lèi)
比如sun.misc.Unsafe,原本只想讓oracle jdk team來(lái)使用,不過(guò)由于這些類(lèi)應(yīng)用太廣泛了,為了向后兼容,java9做了妥協(xié),只是將這些類(lèi)歸到了jdk.unsupported模塊,并沒(méi)有限定其可讀性。
➜ ~ java -d jdk.unsupported jdk.unsupported@9 exports com.sun.nio.file exports sun.misc exports sun.reflect requires java.base mandated opens sun.misc opens sun.reflect
刪除的類(lèi)
java9刪除了sun.misc.BASE64Encoder,這種情況只能改用其他api,比如java.util.Base64
classpath vs module-path
java9引入了模塊系統(tǒng),同時(shí)自身的jdk也模塊化了,引入了module-path,來(lái)屏蔽classpath,也就是說(shuō)在java9優(yōu)先使用module-path,畢竟jdk本身都模塊化了,應(yīng)用本身沒(méi)有模塊化的話,java9通過(guò)unnamed modules及automatic modules機(jī)制來(lái)隱式模塊化,當(dāng)然classpath在java9上還能繼續(xù)使用,比如配合module-path使用等。
沒(méi)有模塊化的jar在classpath會(huì)被歸到unnamed modules;在module-path則會(huì)被自動(dòng)創(chuàng)建為automatic modules(一個(gè)automatic modules會(huì)聲明transitive依賴所有named和unnamed module,然后導(dǎo)出自身的package)
一個(gè)包名不能在多個(gè)模塊中出現(xiàn)(split packages)
因?yàn)槟K中可以exports指定包給其他模塊,如果多個(gè)模塊exports同樣的包名會(huì)造成混亂,特別若有其他類(lèi)庫(kù)同時(shí)requires這兩個(gè)模塊,就不知道該引用那個(gè)模塊的了。
傳遞依賴
如果一個(gè)模塊的接口參數(shù)或返回類(lèi)型使用了其他模塊的類(lèi),則建議requires transitive它依賴的模塊
小心循環(huán)依賴
在設(shè)計(jì)模塊的時(shí)候,要盡可能考慮到是否會(huì)有循環(huán)依賴的問(wèn)題,如果有則需要重新設(shè)計(jì)
使用services來(lái)實(shí)現(xiàn)optional依賴
services特別適合用來(lái)解耦調(diào)用方與實(shí)現(xiàn)類(lèi)依賴的問(wèn)題,如果接口有多種實(shí)現(xiàn)類(lèi),調(diào)用方不必要requires所有的實(shí)現(xiàn)類(lèi),只需要requires接口即可,使用services類(lèi)型來(lái)加載實(shí)現(xiàn)類(lèi)的實(shí)例。通過(guò)在module-path去動(dòng)態(tài)添加實(shí)現(xiàn)模塊實(shí)現(xiàn)解耦。
模塊版本管理
module-info.java不支持聲明版本號(hào),但是創(chuàng)建jar包的時(shí)候,可以通過(guò)--module-version設(shè)置。不過(guò)模塊系統(tǒng)查找模塊的時(shí)候還是使用模塊名來(lái)查找(如果module-path里頭有多個(gè)重名模塊,則模塊系統(tǒng)知會(huì)使用找到的第一個(gè),自動(dòng)忽略后續(xù)的同名模塊),版本依賴問(wèn)題不在模塊系統(tǒng)解決范疇內(nèi),交由maven之類(lèi)的依賴管理工具去管理。
模塊資源訪問(wèn)
模塊化之后資源文件也收到保護(hù),只能由該模塊去訪問(wèn)本模塊自身的資源文件,如果需要跨模塊訪問(wèn),也必須借助ModuleLayer找到目標(biāo)模塊,再調(diào)用目標(biāo)模塊去加載該模塊的資源文件。
反射的使用
這里涉及到deep reflection問(wèn)題,所謂的deep reflection就是通過(guò)反射去調(diào)用一個(gè)class的非public元素。module-info.java的exports聲明package只是允許該package直接所屬的類(lèi)允許訪問(wèn)其public元素,并不允許反射調(diào)用非public元素。
反射在模塊系統(tǒng)里頭需要特殊聲明才允許使用(使用opens聲明允許deep reflection),這樣就導(dǎo)致很多使用反射的類(lèi)庫(kù)諸如spring,需要額外配置才能遷移到j(luò)ava9。解決方案有兩個(gè):一個(gè)是opens package包名給需要反射的模塊,比如spring.beans等;一個(gè)就是直接opens整個(gè)模塊。
默認(rèn)--illegal-access=permit,同時(shí)該設(shè)置只適用于java9之前的package在java9被不允許訪問(wèn),不適用于java9中新的不允許訪問(wèn)的package.(建議遷移到模塊化系統(tǒng)時(shí)設(shè)置為deny)
不過(guò)就是在模塊系統(tǒng)中包名不一樣就屬于不同的包,沒(méi)有繼承關(guān)系,比如com.service.func1與com.service.func2這兩個(gè)是不同的包,你不能只opens com.service,必須分別指定這樣就導(dǎo)致需要open的的package比較多。因此open整個(gè)module可能更省事一點(diǎn),但也屬于比較粗暴的做法。上面的做法是在原來(lái)module-info.java里頭去做修改,另外一種是在執(zhí)行java或javac的時(shí)候通過(guò)指定的命令來(lái)修改原來(lái)的關(guān)系。比如
java ... --add-opens source-module/source-package=target-module
如果需要導(dǎo)出給unnamed modules,則target-module為ALL-UNNAMED
當(dāng)然如果是新的系統(tǒng),那就不建議使用反射了,可以使用MethodHandles及VarHandles。
常見(jiàn)問(wèn)題和措施
ClassNotFoundException/NoClassDefFoundError
比如javax.xml.bind.JAXBException,JAXB已經(jīng)歸入到j(luò)ava.xml.bind模塊,在java命名后面添加
--add-modules java.xml.bind
如果圖省事,把$JAVA_HOME及所有第三方類(lèi)庫(kù)添加到module-path,然后來(lái)個(gè)
--add-modules ALL-MODULE-PATH
illegal reflective access by xxx to method java.lang.ClassLoader.defineClass
反射原因引起,由于舊系統(tǒng)沒(méi)有module-info,因此在java命名添加參數(shù)加以修改
--add-opens java.base/java.lang=ALL-UNNAMED
確定依賴的模塊
通過(guò)IDE或者jdeps分析
jdeps --class-path 'classes/lib/*' -recursive -summary app.jar
jdeps只是靜態(tài)代碼分析,如果有使用反射用的類(lèi)jdeps分析不出來(lái),需要自己手工requires,如果dependency是optional的,可以requires static
對(duì)模塊單元測(cè)試的可讀性問(wèn)題
如果單元測(cè)試時(shí)單獨(dú)模塊的話,可以在運(yùn)行時(shí)通過(guò)--add-exports或--add-opens來(lái)授予單元測(cè)試模塊對(duì)目標(biāo)模塊的可讀性及反射能力。另外由于split packages問(wèn)題,單元測(cè)試類(lèi)的包名不能跟目標(biāo)模塊包名重復(fù)。原來(lái)maven工程那種test
小結(jié)
可以分兩步走遷移到j(luò)ava9,首先是先不模塊化,只先跑在jdk9上;然后再模塊化。
相關(guān)文章
Java如何按16進(jìn)制發(fā)送和接收TCP指令
這篇文章主要介紹了Java如何按16進(jìn)制發(fā)送和接收TCP指令問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09
SpringBoot 項(xiàng)目使用hutool 工具進(jìn)行 http 接口調(diào)用的處理方
在實(shí)際的開(kāi)發(fā)過(guò)程中一個(gè)互聯(lián)網(wǎng)的項(xiàng)目來(lái)說(shuō) ,有可能會(huì)涉及到調(diào)用外部接口的實(shí)際業(yè)務(wù)場(chǎng)景,下面通過(guò)本文給大家介紹SpringBoot 項(xiàng)目 使用hutool 工具進(jìn)行 http 接口調(diào)用的處理方法,需要的朋友可以參考下2022-06-06
淺談Arrays.asList() 和ArrayList類(lèi)型區(qū)別
下面小編就為大家?guī)?lái)一篇Arrays.asList() 和ArrayList類(lèi)型區(qū)別。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-10-10
在Ubuntu系統(tǒng)下安裝JDK和Tomcat的教程
這篇文章主要介紹了在Ubuntu系統(tǒng)下安裝JDK和Tomcat的教程,這樣便是在Linux系統(tǒng)下搭建完整的Java和JSP開(kāi)發(fā)環(huán)境,需要的朋友可以參考下2015-08-08
SpringMVC轉(zhuǎn)發(fā)與重定向參數(shù)傳遞的實(shí)現(xiàn)詳解
這篇文章主要介紹了SpringMVC轉(zhuǎn)發(fā)與重定向參數(shù)傳遞,對(duì)于重定向,可以通過(guò)FlashMap或RedirectAttributes來(lái)在請(qǐng)求間傳遞數(shù)據(jù),因?yàn)橹囟ㄏ蛏婕皟蓚€(gè)獨(dú)立的HTTP請(qǐng)求,而轉(zhuǎn)發(fā)則在同一請(qǐng)求內(nèi)進(jìn)行,數(shù)據(jù)可以直接通過(guò)HttpServletRequest共享,需要的朋友可以參考下2022-07-07
java實(shí)現(xiàn)HttpClient異步請(qǐng)求資源的方法
這篇文章主要介紹了java實(shí)現(xiàn)HttpClient異步請(qǐng)求資源的方法,實(shí)例分析了java基于http協(xié)議實(shí)現(xiàn)異步請(qǐng)求的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07

