带有匿名类的CGLib Enhancer

时间:2014-07-25 09:03:29

标签: java reflection cglib

我正在使用CGLib代理类。我遇到了一个无法增强匿名类的问题,因为它们没有默认的构造函数。

我的代码如下所示:

Enhancer enhancer = new Enhancer();
enhancer.setCallback(new MethodInterceptor() { .... });
enhancer.setSuperclass(type); // type is the Class I'm trying to enhance
enhancer.create();

但是,如果增强类是匿名的(没有默认构造函数),则抛出以下异常。

java.lang.IllegalArgumentException: Superclass has no null constructors but no arguments were given

有办法解决这个问题吗?

1 个答案:

答案 0 :(得分:1)

JVM的验证程序确保您调用有效的构造函数链。解决这个问题的唯一方法是在启动JVM时通过命令行禁用验证程序,这通常不是你想要做的,因为它会给你的应用程序带来一些不安全感。

因此,您必须调用匿名类的有效构造函数,该构造函数由 cglib 创建的子类模仿。因此,该类的构造函数将其封闭类的实例作为其参数。如果您使用 cglib 创建一个从不调用实际方法的纯代理,您可以考虑将null交给此构造函数,但仍需要选择正确的在提交论证之前的构造函数。

您可以致电:

来实现这一目标
enhancer.create(new Class<?>[] {type.getEnclosingClass()}, new Object[] {null})

上面的调用选择了正确的构造函数,并使用null值初始化匿名类的外部实例字段。

或者,您可以使用内部OpenJDK类ReflectionFactory来创建实例而无需调用任何构造函数。调用enhancer.createClass()后可以执行此操作。这需要额外的准备,因为您需要指定不同的回调。

但是,您可能只想完全删除 cglib 。作为免责声明,我是我想向您推荐的替代图书馆的作者。使用Byte Buddy,您可以更自由地创建类,并且使用它更容易定义逻辑。使用Byte Buddy,你可以写

new ByteBuddy()
  .subclass(type)
  .method(any()).intercept(MethodDelegation.to(new Object() {
     @RuntimeType
     public Object intercept(@SuperCall Callable<?> c, 
                             @Origin Method m, 
                             @AllArguments Object[] a) throws Exception {
       // implement your interception logic
     }
   }).make();

并实现相同的逻辑。只需删除调用拦截所不需要的(带注释的)参数。