带有 UrlClassLoader 的 ServiceLoader 找不到服务

问题描述

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

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

这是Test模块的结构截图:

Test Module

这里是 PluginTest 模块结构的截图:

PluginTest Module

这是 ServiceProvider PluginProvider:

package net.lbflabs;

public abstract class PluginProvider {

    public abstract String getSecretMessage(int message);
}

这是扩展这个类的插件

package net.lbflabs.plugins;

import net.lbflabs.PluginProvider;

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

这里是加载插件的主类:

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$2.next(ServiceLoader.java:1310)
    at java.base/java.util.ServiceLoader$2.next(ServiceLoader.java:1298)
    at java.base/java.util.ServiceLoader$3.next(ServiceLoader.java:1396)
    at net.lbflabs.Main.main(Main.java:19)

Process finished with exit code 1

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

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

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

解决方法

我注意到的一件事是您构造了一个仅包含 1 个类的类加载器。有了这个,它将无法创建适当的类层次结构,并且很可能会失败。而是将当前的 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。它只包含类而不包含 META-INF/services/net.lbflabs.PluginProvider