亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

java9中遷移注意問題的示例分析

發布時間:2021-08-05 10:11:36 來源:億速云 閱讀:107 作者:小新 欄目:編程語言

這篇文章給大家分享的是有關java9中遷移注意問題的示例分析的內容。小編覺得挺實用的,因此分享給大家做個參考,一起跟隨小編過來看看吧。

遷移種類

1、代碼不模塊化,先遷移到jdk9上,好利用jdk9的api

2、代碼同時也模塊化遷移

幾點注意事項

不可讀類

比如sun.security.x509,在java9中歸到java.base模塊中,但是該模塊沒有export該package
可以通過運行的時候添加--add-exports java.base/sun.security.x509=ALL-UNNAMED來修改exports設定

內部類

比如sun.misc.Unsafe,原本只想讓oracle jdk team來使用,不過由于這些類應用太廣泛了,為了向后兼容,java9做了妥協,只是將這些類歸到了jdk.unsupported模塊,并沒有限定其可讀性。

? ~ 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

刪除的類

java9刪除了sun.misc.BASE64Encoder,這種情況只能改用其他api,比如java.util.Base64

classpath vs module-path

java9引入了模塊系統,同時自身的jdk也模塊化了,引入了module-path,來屏蔽classpath,也就是說在java9優先使用module-path,畢竟jdk本身都模塊化了,應用本身沒有模塊化的話,java9通過unnamed modules及automatic modules機制來隱式模塊化,當然classpath在java9上還能繼續使用,比如配合module-path使用等。
沒有模塊化的jar在classpath會被歸到unnamed modules;在module-path則會被自動創建為automatic modules(一個automatic modules會聲明transitive依賴所有named和unnamed module,然后導出自身的package)

一個包名不能在多個模塊中出現(split packages)

因為模塊中可以exports指定包給其他模塊,如果多個模塊exports同樣的包名會造成混亂,特別若有其他類庫同時requires這兩個模塊,就不知道該引用那個模塊的了。

傳遞依賴

如果一個模塊的接口參數或返回類型使用了其他模塊的類,則建議requires transitive它依賴的模塊

小心循環依賴

在設計模塊的時候,要盡可能考慮到是否會有循環依賴的問題,如果有則需要重新設計

使用services來實現optional依賴

services特別適合用來解耦調用方與實現類依賴的問題,如果接口有多種實現類,調用方不必要requires所有的實現類,只需要requires接口即可,使用services類型來加載實現類的實例。通過在module-path去動態添加實現模塊實現解耦。

模塊版本管理

module-info.java不支持聲明版本號,但是創建jar包的時候,可以通過--module-version設置。不過模塊系統查找模塊的時候還是使用模塊名來查找(如果module-path里頭有多個重名模塊,則模塊系統知會使用找到的第一個,自動忽略后續的同名模塊),版本依賴問題不在模塊系統解決范疇內,交由maven之類的依賴管理工具去管理。

模塊資源訪問

模塊化之后資源文件也收到保護,只能由該模塊去訪問本模塊自身的資源文件,如果需要跨模塊訪問,也必須借助ModuleLayer找到目標模塊,再調用目標模塊去加載該模塊的資源文件。

反射的使用

這里涉及到deep reflection問題,所謂的deep reflection就是通過反射去調用一個class的非public元素。module-info.java的exports聲明package只是允許該package直接所屬的類允許訪問其public元素,并不允許反射調用非public元素。

反射在模塊系統里頭需要特殊聲明才允許使用(使用opens聲明允許deep reflection),這樣就導致很多使用反射的類庫諸如spring,需要額外配置才能遷移到java9。解決方案有兩個:一個是opens package包名給需要反射的模塊,比如spring.beans等;一個就是直接opens整個模塊。

默認--illegal-access=permit,同時該設置只適用于java9之前的package在java9被不允許訪問,不適用于java9中新的不允許訪問的package.(建議遷移到模塊化系統時設置為deny)
不過就是在模塊系統中包名不一樣就屬于不同的包,沒有繼承關系,比如com.service.func1與com.service.func2這兩個是不同的包,你不能只opens com.service,必須分別指定這樣就導致需要open的的package比較多。因此open整個module可能更省事一點,但也屬于比較粗暴的做法。

上面的做法是在原來module-info.java里頭去做修改,另外一種是在執行java或javac的時候通過指定的命令來修改原來的關系。比如

java ... --add-opens source-module/source-package=target-module

如果需要導出給unnamed modules,則target-module為ALL-UNNAMED

當然如果是新的系統,那就不建議使用反射了,可以使用MethodHandles及VarHandles。

常見問題和措施

ClassNotFoundException/NoClassDefFoundError

比如javax.xml.bind.JAXBException,JAXB已經歸入到java.xml.bind模塊,在java命名后面添加

--add-modules java.xml.bind

如果圖省事,把$JAVA_HOME及所有第三方類庫添加到module-path,然后來個

--add-modules ALL-MODULE-PATH

illegal reflective access by xxx to method java.lang.ClassLoader.defineClass

反射原因引起,由于舊系統沒有module-info,因此在java命名添加參數加以修改

--add-opens java.base/java.lang=ALL-UNNAMED

確定依賴的模塊

通過IDE或者jdeps分析

jdeps --class-path 'classes/lib/*' -recursive -summary app.jar

jdeps只是靜態代碼分析,如果有使用反射用的類jdeps分析不出來,需要自己手工requires,如果dependency是optional的,可以requires static

對模塊單元測試的可讀性問題

如果單元測試時單獨模塊的話,可以在運行時通過--add-exports或--add-opens來授予單元測試模塊對目標模塊的可讀性及反射能力。另外由于split packages問題,單元測試類的包名不能跟目標模塊包名重復。原來maven工程那種test

感謝各位的閱讀!關于“java9中遷移注意問題的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,讓大家可以學到更多知識,如果覺得文章不錯,可以把它分享出去讓更多的人看到吧!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

龙山县| 辽宁省| 乾安县| 六盘水市| 古浪县| 前郭尔| 酒泉市| 泸水县| 日土县| 成都市| 扎赉特旗| 壤塘县| 合江县| 故城县| 潍坊市| 昆明市| 堆龙德庆县| 新田县| 扶风县| 日喀则市| 庄浪县| 陇川县| 志丹县| 南宫市| 静海县| 精河县| 达州市| 谷城县| 岳阳市| 吉林市| 黄陵县| 浙江省| 林西县| 潮州市| 辽宁省| 黎平县| 宜黄县| 霸州市| 监利县| 永靖县| 东辽县|