使用cglib库进行动态生成代理类时调用了默认改造方法,enhancer.create(),为了省事,把目标类写成了内部类,导致错误:
java.lang.IllegalArgumentException: Superclass has no null constructors but no arguments were given at net.sf.cglib.proxy.Enhancer.emitConstructors(Enhancer.java:721) at net.sf.cglib.proxy.Enhancer.generateClass(Enhancer.java:499) 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)
源码:
public class CglibTest { @Test public void createTest() { Student1 defaultProxy = (Student1)this.createCglibProxy(Student1.class); defaultProxy.print(); } private Object createCglibProxy(Class targetClass) { // 也可以使用Enhancer的静态create方法,不需要new 实例。但这样能更好的控制实例配置。 Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(targetClass); enhancer.setCallback(NoOp.INSTANCE); return enhancer.create(); } public class Student1 { private int age; private String name; public int getAge() { return this.age; } public String getName() { return this.name; } public void print() { System.out.println("age:" + age + ", name:" + name); } } }
2. 问题分析
报错显示创建的时候没有给构造方法参数,明显看上去Student1使用的是默认构造方法,不需要参数啊。为了看究竟,查看Student1的反射方法,构造方法果然有一个参数,类型是CglibTest... 原来是通过构造方法外部类被传入了内部类,这也是为什么内部类可以直接调用外部类的成员了。
详细参考:
http://www.2cto.com/kf/201402/281879.html
要点:
1 编译器自动为内部类添加一个成员变量, 这个成员变量的类型和外部类的类型相同, 这个成员变量就是指向外部类对象的引用;
2 编译器自动为内部类的构造方法添加一个参数, 参数的类型是外部类的类型, 在构造方法内部使用这个参数为1中添加的成员变量赋值;
3 在调用内部类的构造函数初始化内部类对象时, 会默认传入外部类的引用。
3. 问题解决
三种方法:
1. 把Student1定义成static 内部类
2. 把Student1拿到外面单独定义
3. cglib创建时传入外部类对象:
enhancer.create(new Class[]{CglibTest.class}, new Object[]{this});