首页 > 解决方案 > 在 Eclipse 中,modulepath 和 classpath 有什么区别?

问题描述

在 Eclipse 中,modulepath 和 classpath 有什么区别,我应该使用哪一个在lib文件夹中添加 JAR 文件?

为什么 JRE 系统库会出现在 modulepath 中?

标签: javaeclipseclasspathjava-modulemodule-path

解决方案


模块系统对代码的影响主要有以下几点:

  • 一个包只能从一个模块访问(嵌套包被视为单独的,所以即使包java.util在模块java.base中,包java.util.logging也可以在模块中java.logging
  • 您只能访问其他模块的导出包中的公共字段和代码方法。即使使用反射也是如此(即java.lang.reflect.AccessibleObject.setAccessible(boolean)仅适用于同一模块中的代码)

类路径上的所有代码一起存在于“未命名”模块中。模块路径上的所有代码都存在于它们自己的“命名”模块中。

您必须区分两种情况:

  • 如果您不将 module-info.java 添加到您的项目中,您的项目将成为未命名模块的一部分,并且可以看到未命名模块中的所有其他代码,以及根模块java.base中的代码和模块中的代码。java.se基本上这意味着在类路径上写代码,一切仍然像在 Java 8 中一样工作,所以你应该把你的依赖关系放在类路径上。

  • 如果您的项目中有一个 module-info.java,那么您的项目将位于其自己的命名模块中,并且只能看到在java.basemodule-info.java 中使用“requires”子句引用的其他命名模块中的代码。由于命名模块只能通过模块路径找到,因此您应该将依赖项放在模块路径上。这甚至适用于在 Java 9 之前创建的 jars,它将获得一个从 .jar 文件名派生的模块名称(在这种情况下,它们被称为“自动”模块)。

JRE 始终位于模块路径上,因此即使从类路径上的代码也无法访问其内部代码。

有一种特殊情况:如果你的项目中有一个 module-info.java 并且你的项目中有测试代码,你通常不想在module-info.java. 有两种解决方案:

  • 创建一个专用的测试模块。这一直是基于 osgi 的项目的惯例。缺点是您只能在测试中使用公共 API

  • Maven使用的解决方案:将您的测试依赖项放在类路径上。编译测试代码时,Maven 添加命令行选项,允许命名模块中的代码读取未命名模块(通过 module-info.java 无法实现)。

在 Eclipse Oxygen 中,Maven 解决方案是不可能的,因为它不知道哪个代码是测试代码,但这已在即将于 6 月发布的Eclipse Photon (4.8) 版本中实现。您已经可以使用来自http://download.eclipse.org/eclipse/downloads/的(功能完整的)里程碑构建。如果您发现任何错误,请在https://bugs.eclipse.org/bugs/报告它们。


推荐阅读