带有 UrlClassLoader 的 ServiceLoader 找不到服务

ServiceLoader with UrlClassLoader not finding service

我正在尝试使用带有 UrlClassLoader 的 java ServiceLoader 从某个 jar 文件加载插件,但我似乎无法找到我的插件 classes。构建两个模块都可以,但是每当我 运行 下面的代码时,我都会得到一个 java.util.NoSuchElementException,即使文件路径是正确的,我也看不到我搞砸的地方

我必须在我的 IntelliJ 项目中添加模块,Test(加载插件并提供 ServiceProvider 的应用程序)和 PluginTest(要加载的插件)。

这是测试模块结构的截图:

下面是 PluginTest 模块的结构截图:

这里是 ServiceProvider PluginProvider:

package net.lbflabs;

public abstract class PluginProvider {

    public abstract String getSecretMessage(int message);
}

这是扩展此 class 的插件:

package net.lbflabs.plugins;

import net.lbflabs.PluginProvider;

public class PluginClass extends PluginProvider {
    @Override
    public String getSecretMessage(int message) {
        return "Here is my secret.";
    }
}

这里是加载插件的主要class:

package net.lbflabs;

import java.io.File;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ServiceLoader;

public class Main {

    private static String path = "C:/Users/jacke/IdeaProjects/Tests/out/artifacts/PluginTest_jar/PluginTest.jar";

    public static void main(String[] args) throws MalformedURLException {
        System.out.println("Initializing... \nPreparing to load JAR from Path " + path + "...");
        File file = new File(path);
        System.out.println(file.exists()); //Prints true, so file does exist
        URLClassLoader c = new URLClassLoader((new URL[]{file.getAbsoluteFile().toURI().toURL()}));
        ServiceLoader<PluginProvider> loader = ServiceLoader.load(PluginProvider.class, c);
        PluginProvider p = loader.iterator().next(); // Throws the java.util.NoSuchElementException
        System.out.println("Secret message in plugin: " + p.getSecretMessage(1));
    }
}

这是 运行测试模块时的输出:

"C:\Program Files\Java\jdk-14.0.1\bin\java.exe" "-javaagent:C:\Program Files\JetBrains\IntelliJ IDEA Community Edition 2020.3\lib\idea_rt.jar=50330:C:\Program Files\JetBrains\IntelliJ IDEA Community Edition 2020.3\bin" -Dfile.encoding=UTF-8 -classpath C:\Users\jacke\IdeaProjects\Tests\out\production\Tests net.lbflabs.Main
Initializing... 
Preparing to load JAR from Path C:/Users/jacke/IdeaProjects/Tests/out/artifacts/PluginTest_jar/PluginTest.jar...
true
Exception in thread "main" java.util.NoSuchElementException
    at java.base/java.util.ServiceLoader.next(ServiceLoader.java:1310)
    at java.base/java.util.ServiceLoader.next(ServiceLoader.java:1298)
    at java.base/java.util.ServiceLoader.next(ServiceLoader.java:1396)
    at net.lbflabs.Main.main(Main.java:19)

Process finished with exit code 1

我认为 PluginTest 模块中的服务文件命名正确 (net.lbflabs.PluginProvider) 并且包含有效的 class 名称 (net.lbflabs.plugins.PluginClass)。如果它不正确,IntelliJ 可能会发现问题,因为当我输入错字时我会收到警告。

如果有人需要,我可以提供整个项目为 .rar(请告诉我你希望我如何分享它,例如我创建了一个 onedrive link)

PS:我已经问过这个问题 here,但是因为我正在做一个更大的项目并且犹豫是否要共享其中的所有代码(而且因为我得到的唯一答案是不起作用,可能是因为我没有提供足够的信息),我想用一个抛出相同异常的最小可行示例重新发布这个问题,我希望这没问题。

我注意到的一件事是您构建了一个 classloader,它只包含 1 个 class。有了这个,它将无法创建适当的 class 层次结构并且很可能会失败。而是将当前 Classloader 作为父级传递给您的 URLClassLoader.

ClassLoade parent = PluginProvider.class.getClassLoader();
URL[] urls = new URL[] { file.getAbsoluteFile().toURI().toURL()};
URLClassLoader c = new URLClassLoader(urls, parent);

现在应该可以构造出合适的层次结构了。

另一件事是在您的项目结构中,您的 META-INF 目录不在包含项目源代码的 src 目录下。这意味着 Intellij 会将它们排除在罐子之外。所以你基本上是在添加一个 jar 而没有 正确的文件。它只包含 class 而不是 META-INF/services/net.lbflabs.PluginProvider.