我正在尝试使用cglib增强java.util.Date
。它不起作用,我对cglib没有经验,所以我想知道出了什么问题。
例如,增强ArrayList
的以下代码有效:
@Test
public void enhance_ArrayList() {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(ArrayList.class);
enhancer.setCallback(new FixedValue() {
@Override
public Object loadObject() throws Exception {
return "Hello cglib!";
}
});
ArrayList enhanced = (ArrayList)enhancer.create();
}
而以下代码:
@Test
public void enhance_Date() {
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(Date.class);
enhancer.setCallback(new FixedValue() {
@Override
public Object loadObject() throws Exception {
return "Hello cglib!";
}
});
Date enhanced = (Date)enhancer.create();
}
导致此异常:
java.lang.IllegalArgumentException
at org.objectweb.asm.ClassReader.<init>(Unknown Source)
at org.objectweb.asm.ClassReader.<init>(Unknown Source)
at org.objectweb.asm.ClassReader.<init>(Unknown Source)
at net.sf.cglib.proxy.BridgeMethodResolver.resolveAll(BridgeMethodResolver.java:61)
at net.sf.cglib.proxy.Enhancer.emitMethods(Enhancer.java:911)
at net.sf.cglib.proxy.Enhancer.generateClass(Enhancer.java:498)
at net.sf.cglib.core.DefaultGeneratorStrategy.generate(DefaultGeneratorStrategy.java:25)
at net.sf.cglib.core.AbstractClassGenerator.create(AbstractClassGenerator.java:216)
at net.sf.cglib.proxy.Enhancer.createHelper(Enhancer.java:377)
at net.sf.cglib.proxy.Enhancer.create(Enhancer.java:285)
答案 0 :(得分:6)
似乎您在版本8中使用的是JDK,其中包含版本8中的类文件.cglib不支持这些类文件,因为此库依赖于过时版本的ASM。
要对此进行调试,我们必须注意ASM不包含任何调试信息,并且不会在其堆栈跟踪中提供所有信息。我们所知道的是,IllegalArgumentException
的构造函数(名为<init>
)抛出了ClassReader
。查看源代码可以发现这种异常只有一种可能性。从ASM 4.2的源代码used by the latest version of cglib,我们可以看到只有在类文件是ASM版本未知的版本时才抛出这样的异常:
// checks the class version
if (readShort(off + 6) > Opcodes.V1_7) {
throw new IllegalArgumentException();
}
不幸的是,没有为此错误提供短信,没有真正原因导致为什么不是这种情况,但我们必须忍受这一点。要修复此错误,您需要一个cglib版本,该版本依赖于支持Java 8的ASM 5+。
截至今天,no compatible version of cglib available因为cglib是not really maintained anymore。您可能想要尝试alternative library such as Byte Buddy(注意我写了这个库,无耻插件)。增强功能可以这样:
new ByteBuddy().subclass(Date.class)
.method(named("toString"))
.intercept(FixedValue.value("Hello world!"))
.make()
.load(getClass().getClassLoader(), ClassLoadingStrategy.Default.WRAPPER)
.getLoaded()
.newInstance();
会覆盖toString
方法,因为Byte Buddy不允许您定义具有非法返回值的类。
答案 1 :(得分:2)
在cglib中有一个issue,由Rafael Winterhalter在他的回答中指出,这阻止了cglib使用winth java 8类。它已在cglib 3.2.0中修复。更新到最新版本(3.2.4)可以解决问题。