内部类引发的cglib创建失败

阅读更多
1. 问题描述
使用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});

你可能感兴趣的:(cglib)