java - 具有匿名类的 CGLib 增强器

标签 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

有办法解决这个问题吗?

最佳答案

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

因此,您必须调用匿名类的有效构造函数,该类是由 cglib 创建的子类模仿的。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();

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

关于java - 具有匿名类的 CGLib 增强器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24951944/

相关文章:

java - 使用Castor根据属性值解析xml

java - getDeclaredConstructor 找不到以接口(interface)作为参数的构造函数

c# - 获取提供给泛型方法的泛型参数类型和值

java - 带接口(interface)的 CGLIB 代理

java - Java stacktrace 中的 $$ 和 <generated> 是什么意思?

java - 如何使用 byte buddy 在运行时有效地将接口(interface)与 POJO 关联?

java - 如何确定Java JDK版本和sdk版本?

Javafx 文本框不工作

java - 为什么使用双重的for循环无法终止

c# - 如何获取泛型方法的 MethodInfo?