Загрузка плагина для пользовательского приложения приводит к возникновению ошибки NoClassDefFoundError - PullRequest
4 голосов
/ 24 ноября 2010

У меня есть некоторые проблемы, связанные с концепцией загрузки классов, я уже некоторое время программирую, но я относительно знаю, как работает загрузка классов, я просмотрел пару примеров и прочитал о деталях, лежащих в основе загрузки классов и самих классов, хотя в некоторой степени я понимаю, что существует концепция, которая ускользает от меня и кажется довольно трудной для выражения в удобных для поиска терминах.

По сути, я пытаюсь создать «гильдии» для игрового режима, который я разрабатывал для Minecraft, эти гильдии лежат в своих собственных классах и загружаются в игру при запуске или всякий раз, когда метод 'reloadGuildFiles ()' выпущен. Я разрабатываю эти классы, сначала экспортируя основное приложение и добавляя его в путь к классам создаваемой гильдии, а также в зависимости от основных приложений.

Вот метод reloadGuildFiles.

public void reloadGuildFiles() {
  unloadGuildFiles();

  synchronized ( _sync ) {
   System.out.println( "Loading guild class files." );

   File guildDataSourceDirectory = new File( "Prospect/Guilds/" );

   URLClassLoader urlcl = null;

   try {
    urlcl = URLClassLoader.newInstance( new URL[] { guildDataSourceDirectory.toURI().toURL() }, Thread.currentThread().getContextClassLoader() );
   } catch ( Exception e ) {
    e.printStackTrace();
    return;
   }

   if ( urlcl == null )
    return;

   for ( File guildDataFile : guildDataSourceDirectory.listFiles() ) {
    if ( !guildDataFile.getName().endsWith( ".class" ) ) {
     System.out.println( "Skipping " + guildDataFile.getName() );
     continue;
    }

    try {
     String className = guildDataFile.getName().substring( 0, guildDataFile.getName().lastIndexOf( "." ) );

     System.out.println( "Loading: " + className + "\n" +
       "\tfrom: " + guildDataFile.getPath() );

     Class<?> clazz = urlcl.loadClass( className );

     Object object = clazz.newInstance();

     if ( object instanceof Guild == false ) {
      System.out.println( "Object loaded is not an instance of Guild." );
      continue;
     }

     Guild guild = ( Guild ) object;

     if ( _guildMap.containsKey( guild.getName() ) ) {
      System.out.println( "Duplicate guild names in guild map: " + guild.getName() );
      continue;
     }

     _guildMap.put( guild.getName(), guild );
     guild.onGuildLoaded();
    } catch ( Exception e ) {
     System.out.println( e.getMessage() );
     e.printStackTrace();
     continue;
    }
   }
  }
 }
}

Вот класс Гильдии, содержащийся в основном приложении.

public abstract class Guild {
 public abstract String getName();

 public void onGuildLoaded() {
  System.out.println( "Loaded: " + getName() );
 }
}

Вот класс, который я пытаюсь загрузить

public class Warrior extends Guild {
 public String getName() {
  returns "Warrior";
 }
}

Вот ошибка, которую он мне дает:

java.lang.NoClassDefFoundError: Guild
        at java.lang.ClassLoader.defineClass1(Native Method)
        at java.lang.ClassLoader.defineClassCond(Unknown Source)
        at java.lang.ClassLoader.defineClass(Unknown Source)
        at java.security.SecureClassLoader.defineClass(Unknown Source)
        at java.net.URLClassLoader.defineClass(Unknown Source)
        at java.net.URLClassLoader.access$000(Unknown Source)
        at java.net.URLClassLoader$1.run(Unknown Source)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at java.net.FactoryURLClassLoader.loadClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at GuildManager.reloadGuildFiles(GuildManager.java:53)
        at Prospect.enable(Prospect.java:64)
        at PluginLoader.load(PluginLoader.java:205)
        at PluginLoader.reloadPlugin(PluginLoader.java:189)
        at je.d(je.java:1196)
        at je.a(je.java:430)
        at bg.a(SourceFile:24)
        at bh.a(SourceFile:218)
        at je.a(je.java:56)
        at dp.a(SourceFile:85)
        at net.minecraft.server.MinecraftServer.h(SourceFile:267)
        at net.minecraft.server.MinecraftServer.run(SourceFile:208)
        at bw.run(SourceFile:482)
Caused by: java.lang.ClassNotFoundException: Guild
        at java.net.URLClassLoader$1.run(Unknown Source)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at java.net.FactoryURLClassLoader.loadClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        ... 25 more

Из того, что я могу собрать и из того, что я понимаю, даже если основное приложение находится на пути сборки загружаемого класса, загрузчик классов не распознает класс Guild. Я думаю, что мне нужно попытаться заставить загрузчик классов распознавать класс Guild, содержащийся в основном приложении. Есть ли способ сделать это или я что-то явно не так делаю?

1 Ответ

1 голос
/ 24 ноября 2010

Единственная возможная проблема, которую я вижу здесь, заключается в том, что Thread.currentThread().getContextClassLoader() по какой-то причине создает загрузчик классов, который нельзя использовать для доступа к Guild классу.

Попробуйте вместо этого:

urlcl = URLClassLoader.newInstance( new URL[] { guildDataSourceDirectory.toURI().toURL() }, Guild.class.getClassLoader()); 
...