我尝试通过覆盖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的原因
感谢阅读和希望即将推出的解决方案。
答案 0 :(得分:0)
首先,当你覆盖一个方法时,它会被优先写在“@Override”之前。像Eclipse这样的一些程序会告诉你覆盖是否不正确(如果你没有遵守一些最重要的规则)
其次,您的类是受保护的,这意味着它只会在它实现的包中看到。你确定覆盖类与这个类在同一个包中吗?也许是你得到这样一个错误的原因。
如果您发布更多代码,则更有可能获得帮助。通过这种方式,也许有人能够准确地告诉你问题所在。
答案 1 :(得分:0)
通常,ClassNotFoundException意味着找不到当前类,NoClassDefFoundError意味着找不到当前加载的类的依赖类(请参阅此stackoverflow thread)。可能你的替换类有一个依赖关系,它不存在于类路径中。