NoClassDefFoundError(更改类名)

时间:2013-07-25 19:51:51

标签: java noclassdeffounderror urlclassloader

我尝试通过覆盖loadClass(String,Boolean)方法来更改类名。

我正在创建一个BukkitPlugin。可以找到Bukkitsource here

类加载器本身工作正常,我测试了它并且所有类都运行良好,在我开始更改类名后错误开始发生。

以下是方法:

import java.io.File;
import java.lang.reflect.Field;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.HashMap;

import org.bukkit.plugin.java.JavaPlugin;

public class PluginClassLoader extends URLClassLoader {

    private final HashMap<String, String> replace;

    public PluginClassLoader(JavaPlugin p, HashMap<String, String> replace) throws Exception {
        super(new URL[0], p.getClass().getClassLoader());
        this.replace = replace;
        File f = null;
        Field file = JavaPlugin.class.getDeclaredField("file");
        file.setAccessible(true);
        f = (File) file.get(p);
        addURL(f.toURI().toURL());
    }

    private final HashMap<String, Class<?>> classes = new HashMap<>();

    @Override
    public Class<?> loadClass(String name) throws ClassNotFoundException {
        String s = replace.get(name);
        if(s != null)
            name = s;
        Class<?> c;
        try {
            c = findClass(name);
        } catch (Exception e) {
            c = super.loadClass(name);
        }
        return c;
    }




    @Override
    protected Class<?> loadClass(String name, boolean b)
        throws ClassNotFoundException {
        String s = replace.get(name);
        if(s != null)
            name = s;
        Class<?> c;
        try {
            c = findClass(name);
        } catch (ClassNotFoundException e) {
            c = super.loadClass(name, b);
        }
        return c;
    }

    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
        if ((name.startsWith("org.bukkit."))
                || (name.startsWith("net.minecraft."))) {
            throw new ClassNotFoundException(name);
        }

        Class<?> result = classes.get(name);

        if (result == null) {
            result = super.findClass(name);
            classes.put(name, result);
        }
        return result;

    }
}

当我使用类加载器时,每次使用其中一个替换类时,我都会得到一个NoClassDefFoundError。我最好的猜测是我必须覆盖另一种方法,但我不知道它可能是什么。

错误日志:

java.lang.NoClassDefFoundError: org/bukkit/craftbukkit/v1_5_R3/CraftServer
at me.leo.itemeffects.IEPlugin.onEnable(IEPlugin.java:25)
at org.bukkit.plugin.java.JavaPlugin.setEnabled(JavaPlugin.java:217)
at org.bukkit.plugin.java.JavaPluginLoader.enablePlugin(JavaPluginLoader.java:457)
at org.bukkit.plugin.SimplePluginManager.enablePlugin(SimplePluginManager.java:381)
at org.bukkit.craftbukkit.v1_6_R2.CraftServer.loadPlugin(CraftServer.java:282)
at org.bukkit.craftbukkit.v1_6_R2.CraftServer.enablePlugins(CraftServer.java:264)
at net.minecraft.server.v1_6_R2.MinecraftServer.l(MinecraftServer.java:313)
at net.minecraft.server.v1_6_R2.MinecraftServer.f(MinecraftServer.java:290)
at net.minecraft.server.v1_6_R2.MinecraftServer.a(MinecraftServer.java:250)
at net.minecraft.server.v1_6_R2.DedicatedServer.init(DedicatedServer.java:151)
at net.minecraft.server.v1_6_R2.MinecraftServer.run(MinecraftServer.java:391)
at net.minecraft.server.v1_6_R2.ThreadServerApplication.run(SourceFile:582)

IEPlugin.java:25就是这样:

CraftServer server = (CraftServer) Bukkit.getServer();

jvm正在搜索错误的classDefinition,它会查找包含'v1_5_R3'的那个,但是我将loadClass()中的类名替换为'v1_6_R2'

我正在用'org.bukkit.craftbukkit.v1_6_R2.CraftServer'替换classname'org.bukkit.craftbukkit.v1_5_R3.CraftServer

包含v1_5_R3的类不存在,这就是我在loadClass(String,Boolean)方法中将其更改为v1_6_R2的原因

感谢阅读和希望即将推出的解决方案。

2 个答案:

答案 0 :(得分:0)

首先,当你覆盖一个方法时,它会被优先写在“@Override”之前。像Eclipse这样的一些程序会告诉你覆盖是否不正确(如果你没有遵守一些最重要的规则)

其次,您的类是受保护的,这意味着它只会在它实现的包中看到。你确定覆盖类与这个类在同一个包中吗?也许是你得到这样一个错误的原因。

如果您发布更多代码,则更有可能获得帮助。通过这种方式,也许有人能够准确地告诉你问题所在。

答案 1 :(得分:0)

通常,ClassNotFoundException意味着找不到当前类,NoClassDefFoundError意味着找不到当前加载的类的依赖类(请参阅此stackoverflow thread)。可能你的替换类有一个依赖关系,它不存在于类路径中。