如何在运行时扩展模块路径

How to expand the module path at runtime

我喜欢服务。我也喜欢模块系统。对我来说不幸的是,在我使用 Java 9 之前,我养成了通过 URLClassLoader 从运行时加载的 jar 中获取服务提供者的习惯,就像这样(我将使用 Java 10 的 var 为简洁起见):

var url = new File("myjar.jar").toURI().toURL();
var cl = new URLClassLoader(new URL[] {url}, getClass().getClassLoader());
var services = ServiceLoader.load(MyService.class, cl);
for (var service : services) {
  ...
}

这工作正常,即使在 Java 9 及更高版本中也是如此,但它会将 jar 加载到类路径中,这意味着它使用旧的 META-INF\services 方法来查找服务提供者。我宁愿使用 module-info 方法,但这需要将 jar 加载到模块路径上,但我找不到任何方法可以做到这一点。所以我在这里,希望这里有人更透彻地了解模块系统会告诉我如何做到这一点(或者不能,如果是这样的话)。

我能做到的最小工作示例 assemble,是

var path = Path.of("myjar.jar");
var cl = new URLClassLoader(new URL[]{path.toUri().toURL()});
var mf = ModuleFinder.of(path);
var cfg = Configuration.resolve(mf, List.of(ModuleLayer.boot().configuration()), mf, Set.of());
var ml = ModuleLayer.defineModulesWithOneLoader(cfg, List.of(ModuleLayer.boot()), cl).layer();
var services = ServiceLoader.load(ml, MyService.class);
services.forEach(System.out::println);

假设 myjar.jar 是一个模块化 jar,声明提供 MyService 实现。

原来我使用了错误的命令行选项(我没有意识到我不应该将 java -jar 与模块化 jar 一起使用)。使用正确的命令,@Holger 的答案有效,除了传递给 Configuration.resolve 的集合需要包含要加载的模块的所有名称,这很容易修复:

var path = Path.of("myjar.jar");
var cl = new URLClassLoader(new URL[]{path.toUri().toURL()});
var mf = ModuleFinder.of(path);
var cfg = Configuration.resolve(mf, List.of(ModuleLayer.boot().configuration()), mf, mf.findAll().stream().map(module -> module.descriptor().name()).collect(Collectors.toSet()));
var ml = ModuleLayer.defineModulesWithOneLoader(cfg, List.of(ModuleLayer.boot()), cl).layer();
var services = ServiceLoader.load(ml, MyService.class);
services.forEach(System.out::println);