Why is --add-modules necessary for modules which are on the module path?(为什么模块路径上的模块必须使用--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?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:为什么模块路径上的模块必须使用--Add-Models?
- Jersey REST 客户端:发布多部分数据 2022-01-01
- value & 是什么意思?0xff 在 Java 中做什么? 2022-01-01
- 将log4j 1.2配置转换为log4j 2配置 2022-01-01
- 从 finally 块返回时 Java 的奇怪行为 2022-01-01
- Spring Boot连接到使用仲裁器运行的MongoDB副本集 2022-01-01
- 如何使用WebFilter实现授权头检查 2022-01-01
- Java包名称中单词分隔符的约定是什么? 2022-01-01
- Safepoint+stats 日志,输出 JDK12 中没有 vmop 操作 2022-01-01
- Eclipse 插件更新错误日志在哪里? 2022-01-01
- C++ 和 Java 进程之间的共享内存 2022-01-01