Java - class 加载导致父 class 的 ClassNotFoundException

Java - class loading results in ClassNotFoundException for parent class

我目前正在为 Java 中的一个程序制作一个小的插件框架。 基本上 GRPluginManager.loadPlugins() 遍历目录中的所有 jar 文件,然后从 jar 中的特定文件中读取此插件的 Main-class 的名称并将其加载到 classpath .之后它调用此插件-Main-class(扩展GRPlugin)的enable()方法。

但是,它不起作用。看起来插件-Main-class 的加载工作正常,只有父 class (GRPlugin) 找不到。但它确实存在。

GRPlugin.java

package io.github.grengine.core.plugins;

public class GRPlugin {

    public void enable(){

    }

    public void disable(){

    }
}

GRPluginManager.java

package io.github.grengine.core.plugins;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.Map;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;

import org.bukkit.plugin.java.JavaPluginLoader;
import org.yaml.snakeyaml.Yaml;

import io.github.grengine.JarUtils;
import io.github.grengine.Log;
import io.github.grengine.Main;

public class GRPluginManager {

    private static final String PLUGIN_PATH = Main.main.getDataFolder()+File.separator+"storage"+File.separator+"plugins";
    private ArrayList<GRPlugin> plugins = new ArrayList<GRPlugin>();

    public ArrayList<GRPlugin> loadPlugins() throws Exception{

    File filePath = new File(PLUGIN_PATH);
    File files [] = filePath.listFiles();

    //Iterate over files in the plugin directory

    for(File file:files){
        if(file.isFile()){

            Log.Info("Reading "+file);

            ZipFile zipFile = new ZipFile(file);


            String fullyQualifiedName = null;
            String plname = null;
            String plversion = null;

            Enumeration<? extends ZipEntry> entries = zipFile.entries();

            while (entries.hasMoreElements()) {
                ZipEntry ze = (ZipEntry) entries.nextElement();
                if(ze.getName().equals("gre"+File.separator+"ext.yml")){
                    Yaml yaml = new Yaml();
                    InputStream is = zipFile.getInputStream(ze);
                    Map<?, ?> map = (Map<?, ?>) yaml.load(is);

                    fullyQualifiedName = (String) map.get("main");
                    plname = (String) map.get("name");
                    plversion = (String) map.get("version");
                }
            }


            zipFile.close();


            Log.Info("Enabling "+plname+" "+plversion+" ...");

            URLClassLoader classLoader = URLClassLoader.newInstance(new URL[] { new URL("file:"+file) });


            Class<?> clazz = classLoader.loadClass(fullyQualifiedName);
            GRPlugin pl = (GRPlugin) clazz.newInstance();
            pl.enable();

Log.Info("Enabling "+plname+" "+plversion+" ...");

        }else {
         //skip folders
         continue;
       }
    }
    return plugins;
    }

}

和插件的主要class(class扩展GRPlugin)/Main.java

package io.github.plutos;

import io.github.grengine.Log;
import io.github.grengine.core.plugins.GRPlugin;

public class Main extends GRPlugin{

    @Override
    public void enable() {
        Log.Info("ENABLED");

    }

    @Override
    public void disable() {
        Log.Info("DISABLED");

    }

}

这是堆栈跟踪:

java.lang.NoClassDefFoundError: io/github/grengine/core/plugins/GRPlugin
        at java.lang.ClassLoader.defineClass1(Native Method) ~[?:1.8.0_05]
        at java.lang.ClassLoader.defineClass(ClassLoader.java:760) ~[?:1.8.0_05]
        at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142) ~[?:1.8.0_05]
        at java.net.URLClassLoader.defineClass(URLClassLoader.java:455) ~[?:1.8.0_05]
        at java.net.URLClassLoader.access0(URLClassLoader.java:73) ~[?:1.8.0_05]
        at java.net.URLClassLoader.run(URLClassLoader.java:367) ~[?:1.8.0_05]
        at java.net.URLClassLoader.run(URLClassLoader.java:361) ~[?:1.8.0_05]
        at java.security.AccessController.doPrivileged(Native Method) ~[?:1.8.0_05]
        at java.net.URLClassLoader.findClass(URLClassLoader.java:360) ~[?:1.8.0_05]
        at java.lang.ClassLoader.loadClass(ClassLoader.java:424) ~[?:1.8.0_05]
        at java.net.FactoryURLClassLoader.loadClass(URLClassLoader.java:798) ~[?:1.8.0_05]
        at java.lang.ClassLoader.loadClass(ClassLoader.java:357) ~[?:1.8.0_05]
        at io.github.grengine.core.plugins.GRPluginManager.loadPlugins(GRPluginManager.java:82) ~[?:?]
        at io.github.grengine.Main.onEnable(Main.java:56) ~[?:?]
        at org.bukkit.plugin.java.JavaPlugin.setEnabled(JavaPlugin.java:321) ~[spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at org.bukkit.plugin.java.JavaPluginLoader.enablePlugin(JavaPluginLoader.java:340) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at org.bukkit.plugin.SimplePluginManager.enablePlugin(SimplePluginManager.java:405) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at org.bukkit.craftbukkit.v1_8_R3.CraftServer.loadPlugin(CraftServer.java:356) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at org.bukkit.craftbukkit.v1_8_R3.CraftServer.enablePlugins(CraftServer.java:316) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at net.minecraft.server.v1_8_R3.MinecraftServer.s(MinecraftServer.java:418) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at net.minecraft.server.v1_8_R3.MinecraftServer.k(MinecraftServer.java:382) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at net.minecraft.server.v1_8_R3.MinecraftServer.a(MinecraftServer.java:337) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at net.minecraft.server.v1_8_R3.DedicatedServer.init(DedicatedServer.java:256) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at net.minecraft.server.v1_8_R3.MinecraftServer.run(MinecraftServer.java:528) [spigot-core.jar:git-Spigot-c5146ba-c637b93]
        at java.lang.Thread.run(Thread.java:745) [?:1.8.0_05]
Caused by: java.lang.ClassNotFoundException: io.github.grengine.core.plugins.GRPlugin
        at java.net.URLClassLoader.run(URLClassLoader.java:372) ~[?:1.8.0_05]
        at java.net.URLClassLoader.run(URLClassLoader.java:361) ~[?:1.8.0_05]
        at java.security.AccessController.doPrivileged(Native Method) ~[?:1.8.0_05]
        at java.net.URLClassLoader.findClass(URLClassLoader.java:360) ~[?:1.8.0_05]
        at java.lang.ClassLoader.loadClass(ClassLoader.java:424) ~[?:1.8.0_05]
        at java.net.FactoryURLClassLoader.loadClass(URLClassLoader.java:798) ~[?:1.8.0_05]
        at java.lang.ClassLoader.loadClass(ClassLoader.java:357) ~[?:1.8.0_05]
        ... 25 more

我只是找不到这个问题的解决方案,我已经尝试了不同的类加载器(ClassLoader.getSystemClassLoader()GRPlugin.class.getClassLoader())...

你能告诉我哪里做错了吗?如果您需要更多信息,请随时询问!

提前致谢, 融合光猫

您可以在ipscan开源项目中看到插件系统的实现。这里是探索的起点:

  • Plugin界面。 ipscan 中的任何插件都必须实现它。
  • PluginLoader class 是您的 GRPluginManager 类比。它从不同的地方(在项目 jar、外部 jar 等)加载插件。在这里查看 PluginClassLoader 的实现,它扩展 URLClassLoader 并从 jar 文件加载 classes。
  • feeders 文件夹,其中包含一些实现了 Plugin 接口的项目内插件。

更新: 你也可以重用你的插件系统标准 java.util.ServiceLoader or some alternative solution

如果您没有为您的自定义类加载器设置父类加载器,系统类加载器将默认用作父类加载器。

在您的环境中,GRPlugin 似乎是由系统类加载器的后代加载的,因此会出现错误。

改用这个:

URLClassLoader classLoader = URLClassLoader.newInstance(new URL[] { new URL("file:"+file) }, GRPlugin.class.getClassLoader());