增强java.util.Date类时,cglib会抛出IllegalArgumentException

我试图用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(); } 

导致此exception:

 java.lang.IllegalArgumentException at org.objectweb.asm.ClassReader.(Unknown Source) at org.objectweb.asm.ClassReader.(Unknown Source) at org.objectweb.asm.ClassReader.(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) 

您似乎在版本8中使用JDK,其中包含版本8中的类文件.cglib不支持这些类文件,因为此库依赖于过时的ASM版本。

要调试它,我们必须注意ASM不包含任何调试信息,并且不在其堆栈跟踪中提供所有信息。 我们所知道的是,从ClassReader的构造函数(名为 )抛出了IllegalArgumentException 。 查看源代码表明,这种exception只有一种可能性。 从最新版本的cglib使用的ASM 4.2的源代码中,我们可以看到,如果类文件是ASM版本未知的版本,则仅抛出此类exception:

 // checks the class version if (readShort(off + 6) > Opcodes.V1_7) { throw new IllegalArgumentException(); } 

不幸的是,没有为此错误提供短信,没有真正的理由说明为什么不是这种情况,但我们必须忍受这一点。 要修复此错误,您需要一个cglib版本,该版本依赖于支持Java 8的ASM 5+。

截至今天, 没有兼容的cglib版本可用,因为cglib 不再真正维护 。 您可能想要尝试替代库,例如Byte Buddy (注意我写了这个库,无耻插件)。 增强function可以是这样的:

 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不允许您定义具有非法返回值的类。

cglib中存在一个问题 ,Rafael Winterhalter在他的回答中指出,这阻止了cglib使用winth java 8类。 它在cglib 3.2.0中得到修复。 更新到最新版本( 3.2.4 )可以解决问题。