8

我面临的问题是 jlink 的服务绑定选项链接了很多很多模块,它们似乎都不是必需的。当省略服务绑定选项时,这些模块不会链接。

问题:

  • Q1:你在你的环境中看到同样的行为吗?
  • Q2:这是错误还是期望的行为?
  • Q3:为什么所有这些模块都链接在一起?

我的应用程序:该应用程序是一个简单的服务,由一个接口、一个提供者和一个消费者组成,每个都打包到一个单独的模块中,称为modServicemodProvidermodConsumer(详细信息如下)。

操作系统:Windows 10

Jlink without --bind-services产生预期结果:

jlink --module-path "mods;%JAVA_HOME%\jmods"
    --add-modules modConsumer
    --output myRuntime

java --list-modules
java.base@9
modConsumer
modService 

当应用该--bind-services选项时,我希望另外模块modProvider应该被链接。但是,看看会发生什么(三个自定义模块在最后):

jlink --module-path "mods;%JAVA_HOME%\jmods"
    --bind-services
    --add-modules modConsumer
    --output myRuntime

java --list-modules
java.base@9
java.compiler@9
java.datatransfer@9
java.desktop@9
java.logging@9
java.management@9
java.management.rmi@9
java.naming@9
java.prefs@9
java.rmi@9
java.scripting@9
java.security.jgss@9
java.security.sasl@9
java.smartcardio@9
java.xml@9
java.xml.crypto@9
jdk.accessibility@9
jdk.charsets@9
jdk.compiler@9
jdk.crypto.cryptoki@9
jdk.crypto.ec@9
jdk.crypto.mscapi@9
jdk.deploy@9
jdk.dynalink@9
jdk.internal.opt@9
jdk.jartool@9
jdk.javadoc@9
jdk.jdeps@9
jdk.jfr@9
jdk.jlink@9
jdk.localedata@9
jdk.management@9
jdk.management.cmm@9
jdk.management.jfr@9
jdk.naming.dns@9
jdk.naming.rmi@9
jdk.scripting.nashorn@9
jdk.security.auth@9
jdk.security.jgss@9
jdk.unsupported@9
jdk.zipfs@9
modConsumer
modProvider
modService

我不知道为什么所有这些模块都被链接,因为提供者只返回一个字符串,因此不需要除 java.base 之外的其他 jdk 模块。

以下是 Java 工件:

package test.service; 

public interface HelloService { 
  public String sayHello(); 
}

package test.provider; 
import test.service; 

public class HelloProvider implements HelloService { 
  @Override public String sayHello() { return "Hello!"; }
}

package test.consumer; 
import test.service; 
import java.util.ServiceLoader; 

public class HelloConsumer { 
  public static void main(String... args) { 
    ServiceLoader.load(HelloService.class).forEach(s -> System.out.println(s.sayHello())); 
  }
}

module modService { 
  exports test.service; 
}

module modProvider { 
  requires modService; 
  provides test.service.HelloService with test.provider.HelloProvider; 
}

module modConsumer { 
  requires modService; 
  uses test.service.HelloService; 
} 

任何帮助表示赞赏。

4

2 回答 2

6

精简版

  • Q1:是的。
  • Q2:期望的行为
  • Q3:因为你jlink是这么说的--bind-services

长版

默认情况下jlink不绑定服务,以保持创建的运行时尽可能小。这会随着 的变化而变化--bind-services文档

链接服务提供者模块及其依赖项。

这反映了常规模块解析期间的行为,在解决所有依赖关系之后,所有提供这些模块使用的服务的模块都包含在可读性图中。

在您的情况下也会发生同样的情况,因此提供java.basemodConsumermodService使用的服务的所有模块都包含在映像中。正如你所发现的那样,这相当多。

解决方案

如果你想避免这种情况,你必须放弃--bind-services,而是明确列出你想在图像中看到的提供者:

jlink --module-path "mods;%JAVA_HOME%\jmods"
    --add-modules modConsumer,modProvider
    --output myRuntime
于 2018-01-26T21:28:01.143 回答
5

jlink文档中所述。这

--bind-services 

链接服务提供者模块及其依赖项。

此外,同一个样本表明,

选项将从根模块解析的模块与服务绑定链接;见 Configuration.resolveAndBind 方法。

在您之前的命令中,根模块和默认情况下在模块图中解析的模块是:

java.base@9
modConsumer
modService

此外,使用--bind-services标志时列出的其他模块通过java.base模块解析。

我希望另外模块modProvider应该被链接

正如nicolai所建议的,您可以添加提供程序模块并确保它也在模块图中得到解决。

 --add-modules modConsumer,modProvider

把想法大声说出来。1. 当前寻找提供者的过程是迭代的。2. 是否可以在明确查找服务提供者的同时指定要查找的模块?

于 2018-01-26T19:37:25.010 回答