我正在使用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
有办法解决这个问题吗?
答案 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();
并实现相同的逻辑。只需删除调用拦截所不需要的(带注释的)参数。