我正在JDK 9上试验SPI。整个示例适用于没有" module-info.java"的JDK 9。添加" module-info.java" ServiceLocator没有找到实现类。我很困惑,我在模块化的JDK 9项目中找不到工作的SPI示例。
所以我的示例项目看起来像这样:
/spidemo
├── apiModule
│ ├── pom.xml
│ └── src
│ └── main
│ └── java
│ ├── eu
│ │ └── com
│ │ └── example
│ │ └── text
│ │ └── spi
│ │ └── TextAPI.java
│ └── module-info.java
├── applicationB
│ ├── pom.xml
│ └── src
│ └── main
│ ├── java
│ │ └── eu
│ │ └── com
│ │ └── example
│ │ └── spi
│ │ └── b
│ │ └── application
│ │ └── DemoB.java
│ └── module-info.java
├── applicationCommon
│ ├── pom.xml
│ └── src
│ └── main
│ └── java
│ ├── eu
│ │ └── com
│ │ └── example
│ │ └── spi
│ │ └── application
│ │ └── TextAPIProvider.java
│ └── module-info.java
├── implementationB
│ ├── pom.xml
│ └── src
│ └── main
│ ├── java
│ │ └── eu
│ │ └── com
│ │ └── example
│ │ └── implb
│ │ └── text
│ │ └── TextB.java
│ ├── module-info.java
│ └── resources
│ └── META-INF
│ └── services
│ └── eu.com.example.text.spi.TextAPI
我已经介绍了界面:
package eu.com.example.text.spi;
public interface TextAPI {
String getHelloWorldText();
}
此界面由以下人员实施:
package eu.com.example.implb.text;
import eu.com.example.text.spi.TextAPI;
public class TextB implements TextAPI {
public String getHelloWorldText() {
return "Text from B implementation";
}
}
通过以下代码搜索实现:
package eu.com.example.spi.application;
import eu.com.example.text.spi.DefaultTextAPI;
import eu.com.example.text.spi.TextAPI;
import java.util.ServiceLoader;
public class TextAPIProvider {
public static TextAPI getProvider(String providerName) {
ServiceLoader<TextAPI> serviceLoader = ServiceLoader.load(TextAPI.class);
for (TextAPI provider : serviceLoader) {
String className = provider.getClass().getName();
if (providerName.equals(className)) {
return provider;
}
}
throw new RuntimeException(providerName + " provider is not found!");
}
}
现在是有趣的部分。当我在下面执行课程时没有:
然后找到实现类并打印出文本。
package eu.com.example.spi.b.application;
import eu.com.example.spi.application.TextAPIProvider;
public class DemoB {
public static void main(String[] args) {
System.out.println("---> " + TextAPIProvider.getProvider("eu.com.example.implb.text.TextB").getHelloWorldText());
}
}
介绍这两个&#34; module-info.java&#34; ServiceLocator找不到文件实现类。 /applicationB/src/main/java/module-info.java的内容:
module eu.com.example.applicationB {
requires eu.com.example.apiModule;
requires transitive eu.com.example.applicationCommon;
uses eu.com.example.text.spi.TextAPI;
}
/implementationB/src/main/java/module-info.java的内容:
module eu.com.example.implb.text {
requires eu.com.example.apiModule;
exports eu.com.example.implb.text;
// provides eu.com.example.implb.text.TextB with eu.com.example.text.spi.TextAPI;
}
当我取消注释时:
provides eu.com.example.implb.text.TextB with eu.com.example.text.spi.TextAPI;
行然后发生编译错误:
.../implementationB/src/main/java/module-info.java:[7,74] the service implementation type must be a subtype of the service interface type, or have a public static no-args method named "provider" returning the service implementation
.../implementationB/src/main/java/module-info.java:[7,5] service implementation must be defined in the same module as the provides directive
我试图更改包名称作为编译错误消息,但后来我介绍了&#34; split package&#34;的问题。
在完全模块化的JDK 9中使用ServiceLocator应该怎么做?可能吗?有没有人见过工作的例子? 代码也可以在这里看到:https://github.com/RadoslawOsinski/spidemo
答案 0 :(得分:2)
您可以更改为使用: -
provides eu.com.example.text.spi.TextAPI with eu.com.example.implb.text.TextB;
// you provide a service through its implementation
而不是
provides eu.com.example.implb.text.TextB with eu.com.example.text.spi.TextAPI;
文档中的Services 提供了有关实施的示例。
答案 1 :(得分:0)
一个模块可以指定它通过特定类型的服务提供者提供服务。它是使用 provides
和 with
关键字声明的。
语法:provides serviceType with implementationTypes;
(可以将多个实现类型指定为逗号分隔列表)
因此在您的模块 eu.com.example.implb.text
中应添加以下语句。
provides eu.com.example.implb.text.TextB with eu.com.example.text.spi.TextAPI;
注意:provides
不等于 exports
。因此,如果不导出,任何其他需要 eu.com.example.implb.text
的模块将无法访问 eu.com.example.implb.text.TextB
。