85

我正在使用 Maven 构建 Eclipse 中的 Java 项目。我正在使用来自旧项目的一些回收代码,其中一个类META-INF/services在 JAR 文件夹中查找具有特定名称的文件,然后解析该文件的文本。在这个特定的示例中,它查找具有 Java 接口名称的文件,然后从文件中获取实现的类名。

所以基本上我要做的是在 JAR 的“META-INF/services”文件夹中包含一个文件,文件名 (X) 和一行文本 (Y)。我猜这应该使用 Maven 来完成,也许是通过在 POM 文件中指定一个属性,但我的研究没有发现任何东西。我知道你不应该硬编码或手动输入任何 META 文件,所以我不确定在这里做什么。

4

3 回答 3

136

使用 location创建一个新的源文件夹src/main/resources,然后在其中创建您的META-INF/services文件夹并放入您的完全限定类名 (FQCN) 文件。这应该会自动将它们复制到 jar 文件中。因此,对于具有 FQCN 的接口的实现com.acme.MyInterface,您将拥有:

Project
| src
| | main
|   | java
|     | [your source code]
|   | resources
|     | META-INF
|       | services
|         | com.acme.MyInterface

请注意,这com.acme.MyInterface是文件的名称,而不是像 Java 包这样的目录结构。该文件的名称是您正在实现的接口的 FQCN,在其中,您将在自己的行上拥有每个实现的 FQCN,例如:

com.example.MyInterfaceImpl
com.example.AnotherMyInterfaceImpl

值得注意的是,这也适用于具有默认源集的 Gradle 项目。

完成此操作后,您可以使用以下命令加载接口的所有实现ServiceLoader

ServiceLoader<MyInterface> loader = ServiceLoader.load(MyInterface.class);
for (MyInterface service : loader) {
    // Prints com.example.MyInterfaceImpl and com.example.AnotherMyInterfaceImpl
    System.out.println(service.class.getName());
}

需要注意的一些事项:

  • 所有的实现都必须有一个无参数的构造函数
  • 使用模块或自定义类加载器的应用程序可能必须使用ServiceLoader.load

如果这些条件不适合您,那么您可能想要切换到另一个系统,例如像 Spring、EJB 等 CDI 风格的框架。

于 2013-07-08T16:38:56.047 回答
25

默认情况下,Maven 在以下位置查找资源:

src/main/resources

所以把它放在

src/main/resources/META-INF/services
于 2013-07-08T16:38:59.517 回答
16

或者,如果您的项目不使用标准目录结构(或者您只是希望拥有备用资源目录),您可以手动指定 POM 文件中的资源目录。

例如,如果您META-INF/services位于resources项目目录根目录中的名为的文件夹中,则可以按如下方式指定它:

<project>
...
  <build>
    ...
    <resources>
      <resource>
        <directory>resources</directory>
      </resource>
    </resources>
    ...
  </build>
  ...
</project>

您可以使用它通过添加多个<resource>元素来指定多个目录。

于 2014-08-01T22:59:45.927 回答