我试图用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)
解决方法
您似乎在版本8中使用JDK,其中包含版本8中的类文件.cglib不支持这些类文件,因为此库依赖于过时的
ASM版本.
要对此进行调试,我们必须注意ASM不包含任何调试信息,并且不在其堆栈跟踪中提供所有信息.我们所知道的是,从其ClassReader的构造函数(名为< init>)抛出了IllegalArgumentException.查看源代码可以发现这种异常只有一种可能性.从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(); }
不幸的是,没有为此错误提供短信,没有真正原因导致为什么不是这种情况,但我们必须忍受这一点.要修复此错误,您需要一个依赖于支持Java 8的ASM 5的cglib版本.
截至今天,有一个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不允许您定义具有非法返回值的类.