java - 使用 Byte Buddy 拦截默认构造函数

标签 java bytecode byte-buddy

我正在尝试使用 Byte Buddy 拦截构造函数调用,这是我的示例代码:

package t;

import static net.bytebuddy.dynamic.loading.ClassLoadingStrategy.Default.INJECTION;
import static net.bytebuddy.implementation.MethodDelegation.to;
import static net.bytebuddy.matcher.ElementMatchers.any;
import java.util.concurrent.Callable;
import net.bytebuddy.ByteBuddy;
import net.bytebuddy.implementation.bind.annotation.RuntimeType;
import net.bytebuddy.implementation.bind.annotation.SuperCall;

public class Bar {
  public static void main (String[] args) {
    new ByteBuddy ().subclass (Object.class)
      .name ("t.Foo").constructor (any ()).intercept (to (new Object () {

        @RuntimeType
        public Object construct (@SuperCall Callable<Object> z) throws Exception {
          System.out.println ("CALLING XTOR");
          return z.call ();
        }
      })).make ().load (Bar.class.getClassLoader (), INJECTION).getLoaded ();
  }
}

这会产生以下异常:

Exception in thread "main" java.lang.IllegalStateException: Error invoking java.lang.ClassLoader#findClass
  at net.bytebuddy.dynamic.loading.ClassInjector$UsingReflection$Dispatcher$Resolved.loadClass(ClassInjector.java:392)
  at net.bytebuddy.dynamic.loading.ClassInjector$UsingReflection.inject(ClassInjector.java:201)
  at net.bytebuddy.dynamic.loading.ClassLoadingStrategy$Default$InjectionDispatcher.load(ClassLoadingStrategy.java:199)
  at net.bytebuddy.dynamic.loading.ClassLoadingStrategy$Default.load(ClassLoadingStrategy.java:114)
  at net.bytebuddy.dynamic.DynamicType$Default$Unloaded.load(DynamicType.java:3302)
  at t.Bar.main(Bar.java:21)
Caused by: java.lang.ClassFormatError: Bad method name at constant pool index 19 in class file t/Foo$auxiliary$EalKkAhD
  at java.lang.ClassLoader.defineClass1(Native Method)
  at java.lang.ClassLoader.defineClass(ClassLoader.java:760)
  at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
  at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
  at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
  at java.lang.reflect.Method.invoke(Method.java:497)
  at net.bytebuddy.dynamic.loading.ClassInjector$UsingReflection$Dispatcher$Resolved.loadClass(ClassInjector.java:388)
  ... 5 more

我看到了这个related question然而,我在那里得到了一个碰撞异常(好像构造函数被定义了两次)。

最佳答案

您不能为构造函数的 super 方法调用创建处理程序。 JVM 的 validator 确保任何构造函数调用另一个构造函数一次且仅一次。此调用必须硬编码到调用构造函数的方法中。

请注意,此限制适用于任何 Java 字节码,甚至适用于处理 MethodHandle 的场景。构造函数是个大异常(exception)。

要使您的示例工作,您需要在委托(delegate)之前或之后调用 super 构造函数。请注意,您不能从拦截器访问由 this 定义的任何属性(例如,通过 @This@SuperCall,但是可以阅读参数)如果你拦截 before 调用另一个构造函数。

对于您的代码示例,只需将拦截器与 SuperCall 链接起来,这样调用就会变成硬编码,如以下示例代码所示:

new ByteBuddy().subclass(Object.class)
               .name("t.Foo")
               .constructor(any()).intercept(to(new Object() {
                 public void construct() throws Exception {
                   System.out.println("CALLING XTOR");
                 }
               }).andThen(SuperMethodCall.INSTANCE)) // This makes the difference!
               .make()
               .load(Bar.class.getClassLoader(), INJECTION)
               .getLoaded()

我同意 Byte Buddy 应该提供有关此问题的更多信息。我将为这种情况添加适当的运行时处理(即不再考虑绑定(bind)该方法)。

关于java - 使用 Byte Buddy 拦截默认构造函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34907382/

相关文章:

java - 带有 EX、EEX、EXP 或 EE 科学记数法的字符串要加倍

java - AspectJ,没有构造函数的通用切入点

Java 字节码可视化工具

Python字节码和.pyc文件格式规范

java - 在 jfreechart TimeSeriesCollection 中如何显示仅具有值的日期?

java - 如何使用Byte Buddy更改字节码?

byte-buddy - ByteBuddy 中是否有任何方法可以将 TypeDescription.Generic 转换为适当的 java.lang.reflect.Type?

java - 为什么 Bytebuddy 不能检测 org.slf4j 类?

java - 如何使用bytebuddy生成pojo代码

java - Spring Hibernate 中的多个数据库