问题描述
我正在尝试使用带有 urlclassloader 的 java ServiceLoader 从某个 jar 文件加载插件,但我似乎无法让它找到我的插件类。构建两个模块都有效,但是每当我运行下面的代码时,我都会得到一个 java.util.NoSuchElementException
,即使文件路径是正确的并且我看不到我搞砸的地方
我必须在我的 IntelliJ 项目中使用模块,Test(加载插件并提供 ServiceProvider 的应用程序)和 PluginTest(要加载的插件)。
这是Test模块的结构截图:
这里是 PluginTest 模块结构的截图:
这是 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
。