本文介紹了為什么模塊路徑上的模塊必須使用–Add-Models?的處理方法,對大家解決問題具有一定的參考價值,需要的朋友們下面隨著小編來一起學習吧!
問題描述
一個例子:自從從JDK中刪除了JavaFX,現在JavaFX SDK以一組模塊化JAR的形式分發。要編譯一個JavaFX應用程序,當然必須將它們放在模塊路徑上:
javac -p /path/to/jars/ App.java
然而,這還不夠。嘗試編譯將導致許多類似
的錯誤
sample/App.java:3: error: package javafx.application is not visible
import javafx.application.Application;
^
(package javafx.application is declared in module javafx.graphics, which is not in the module graph)
要解決此問題,我們可以使用–Add-MODULES:
添加
javac -p /path/to/jars/ --add-modules javafx.graphics App.java
如果我們向項目中添加了一個-info.Java模塊(僅包含module ui {}
),則不會有任何問題。
為什么模塊路徑上的模塊對命名模塊可見,但對未命名模塊不可見?
推薦答案
+1回答一個非常好的問題。這里的問題是,當您編譯命名模塊和未命名模塊時,它們的默認根模塊集的計算方式非常不同。
這是JEP 261中的一句話,它解釋了這種差異:
當編譯器編譯未命名模塊中的代碼或Java
調用Launcher并加載應用程序的主類
從類路徑到應用程序類的未命名模塊
加載器,則未命名模塊的缺省根模塊集為
計算方法如下:如果存在,則java.se模塊是根。如果它不存在,那么
升級模塊路徑上或系統之間的每個Java.*模塊
不加限定地導出至少一個包的模塊是
超級用戶。升級模塊路徑上或系統間的每個非Java.*模塊
不加限定地導出至少一個包的模塊是
也是根。
這看起來可能有點復雜,所以我用粗體顯示了文本中最重要的部分。另外,讓我們一步一步來:
您沒有module-info.java
,所以您的模塊是一個未命名的模塊。
java.se
存在,因此它已進入根集。
您的升級模塊路徑為空(因為您只指定了-p
,而沒有指定--upgrade-module-path
)。
至少導出一個包的系統模塊也進入集合。
因此,根集僅為java.se
和一些系統模塊。沒有JavaFX模塊進入集合!
現在,當您使用module-info.java
編譯時會發生什么?使用不同的規則計算根集:
否則,默認的根模塊集取決于階段:
在編譯時,通常是正在編譯的模塊集
因為根模塊是需要JavaFX模塊的模塊,所以它們進入了模塊圖。
那么,如何解決這個問題呢?您可以通過將JavaFX模塊放在升級模塊路徑上來完成此操作:
javac --upgrade-module-path /path/to/jars/ App.java
或使用--add-modules
:
javac -p /path/to/jars/ --add-modules ...
或使用普通老類路徑:
javac -cp /path/to/jars/ App.java
這三個選項都應該有效。讓我知道第一個選項是否真的有效,因為我沒有嘗試它。
這篇關于為什么模塊路徑上的模塊必須使用–Add-Models?的文章就介紹到這了,希望我們推薦的答案對大家有所幫助,