1

所以,我目前正在为我的命令行“操作系统”开发一个插件系统(插件是位于插件文件夹中的 jar 文件)。工作正常,但 API 没有。

我正在使用注释来获取插件主类和事件类。但是,它似乎没有找到我的注释。

这是我用来加载我的插件的:

public static void loadPlugin(File file)
{
    try
    {
        URL urlList[] =
        { new File("plugins/" + file.getName()).toURI().toURL() };
        URLClassLoader classLoader = new URLClassLoader(urlList);
        Class<?> pluginclass, eventclass = pluginclass = null;
        String name, version = name = null;
        Priority priority = Priority.NORMAL;

        String[] classes = Utils.getJarClasses(file, "plugin").toArray(new String[]
        {});

        for (String s : classes)
        {
            System.out.println(s);
            Class<?> c = classLoader.loadClass(s);
            Plugin p = c.getAnnotation(Plugin.class);
            EventHandler e = c.getAnnotation(EventHandler.class);

            if (p != null)
            {
                System.out.println("not null!");
                pluginclass = c;
                name = p.name();
                version = p.version();
                priority = p.priority();
            }

            if (e != null)
            {
                eventclass = c;
            }
        }

        switch (priority)
        {
        case NORMAL:
            plugins.add(new Class<?>[]
            { pluginclass, eventclass });
        case DEVELOPER_API:
            apis.add(new Class<?>[]
            { pluginclass, eventclass });
        }
        print("Loaded Plugin: " + name + " V" + version + " with priority " + priority.toString() + ".");
    } catch (Exception e)
    {
        e.printStackTrace();
    }
}

它正在打印出它当前循环的类,但是pe始终为空,无论如何。显然,版本名称也始终为空。

我的注释类有@Retention(RetentionPolicy.RUNTIME),这就是我在这里问的主要原因。

我测试了一个类似的系统,只是没有使用 jar 文件(而是包内的类)并且工作正常。

提前致谢。

4

1 回答 1

3

可以解释所描述行为的一个原因是您正在混合类加载器:只有当注释类和带注释的类型都由同一个类加载器加载时,注释才会可见。这与使用反射 API 作为实际执行注释查找过程所采用的机制有关。

在您的代码中,您使用自定义类加载器 ,classLoader从外部 jar 文件加载类,但是Plugin和类都由运行您的方法EventHandler的类加载器加载。loadPlugin

为了解决这个问题,我认为您可以提供运行该loadPlugin方法的类加载器作为自定义类加载器的父类加载器,例如:

URLClassLoader classLoader = new URLClassLoader(urlList, YourClass.class.getClassLoader());

以前在 stackoverflow 中也曾报道过类似的行为。例如,参见this questionthis other one的公认答案。

于 2020-11-09T19:03:18.813 回答