48

我正在尝试为我的应用程序创建一个插件系统,我想从一些简单的东西开始。每个插件都应该打包在一个 .jar 文件中并实现SimplePlugin接口:

package plugintest;

public interface SimplePlugin {
    public String getName();
}

现在我已经创建了SimplePlugin一个 .jar 的实现,并把它放在主应用程序的 plugin/ 子目录中:

package plugintest;

public class PluginTest implements SimplePlugin {
    public String getName() {
        return "I'm the plugin!";
    }
}

在主应用程序中,我想获取PluginTest. 我尝试了两种选择,都使用java.util.ServiceLoader.

1.动态扩展类路径

这使用已知的 hack 在系统类加载器上使用反射来避免封装,以便添加URL类路径。

package plugintest.system;

import plugintest.SimplePlugin;

import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Iterator;
import java.util.ServiceLoader;

public class ManagePlugins {
    public static void main(String[] args) throws IOException {
        File loc = new File("plugins");
        extendClasspath(loc);

        ServiceLoader<SimplePlugin> sl = ServiceLoader.load(SimplePlugin.class);
        Iterator<SimplePlugin> apit = sl.iterator();
        while (apit.hasNext())
            System.out.println(apit.next().getName());
    }

    private static void extendClasspath(File dir) throws IOException {
        URLClassLoader sysLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
        URL urls[] = sysLoader.getURLs(), udir = dir.toURI().toURL();
        String udirs = udir.toString();
        for (int i = 0; i < urls.length; i++)
            if (urls[i].toString().equalsIgnoreCase(udirs)) return;
        Class<URLClassLoader> sysClass = URLClassLoader.class;
        try {
            Method method = sysClass.getDeclaredMethod("addURL", new Class[]{URL.class});
            method.setAccessible(true);
            method.invoke(sysLoader, new Object[] {udir});
        } catch (Throwable t) {
            t.printStackTrace();
        }
    }
}

plugins/ 目录按预期添加(可以检查调用sysLoader.getURLs()),但是ServiceLoader对象给出的迭代器为空。

2. 使用 URLClassLoader

这使用了另一个定义,ServiceLoader.load带有类的第二个参数ClassLoader

package plugintest.system;

import plugintest.SimplePlugin;

import java.io.File;
import java.io.FileFilter;
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Iterator;
import java.util.ServiceLoader;

public class ManagePlugins {
    public static void main(String[] args) throws IOException {
        File loc = new File("plugins");

        File[] flist = loc.listFiles(new FileFilter() {
            public boolean accept(File file) {return file.getPath().toLowerCase().endsWith(".jar");}
        });
        URL[] urls = new URL[flist.length];
        for (int i = 0; i < flist.length; i++)
            urls[i] = flist[i].toURI().toURL();
        URLClassLoader ucl = new URLClassLoader(urls);

        ServiceLoader<SimplePlugin> sl = ServiceLoader.load(SimplePlugin.class, ucl);
        Iterator<SimplePlugin> apit = sl.iterator();
        while (apit.hasNext())
            System.out.println(apit.next().getName());
    }
}

再一次,迭代器从来没有“下一个”元素。

我肯定缺少一些东西,因为这是我第一次“玩”类路径和加载。

4

3 回答 3

40

问题很简单。而且很愚蠢。在插件 .jar 文件中,/services/plugintest.SimplePlugin目录中缺少该META-INF文件,因此ServiceLoader无法将 jar 识别为服务并加载类。

差不多就是这样,第二种(更清洁的)方式就像一个魅力。

于 2013-04-19T15:26:13.583 回答
6

从 Java 9 开始,提供扫描的服务将变得更加容易和高效。不再需要META-INF/services.

在接口模块声明中声明:

uses com.foo.spi.Service;

在提供者的模块中:

provides com.foo.spi.Service with com.bar.ServiceImplementation
于 2017-07-21T05:42:54.673 回答
3

Oracle 文档中已经描述了您的应用程序概念的解决方案(包括动态加载 JAR)

使用 Java 平台创建可扩展的应用程序 http://www.oracle.com/technetwork/articles/javase/extensible-137159.html

在文章的底部,您会找到指向

  • 示例的源代码
  • Javadoc 服务加载器 API

在我看来,最好稍微修改 Oracle 的示例,而不是像 Omer Schleifer 所说的那样重新发明轮子。

于 2015-01-28T19:57:57.747 回答