无法获取Byte Buddy拦截器中的方法

问题描述 投票:0回答:1

当我添加时:

@Origin Method method

致我的拦截器:

@RuntimeType
  public static void intercept(
      @Origin Method method, @RuntimeType @AllArguments Object[] allArguments) {
    System.out.println("intercepted with args: " + Arrays.toString(allArguments));
    System.out.println("intercepted method: " + method);
  }

我最终遇到了一个例外:

Error invoking java.lang.instrument.Instrumentation#retransformClasses
class redefinition failed: attempted to change the schema (add/remove fields)

我的设置如下:

ByteBuddyAgent.install();
    final ByteBuddy byteBuddy = new ByteBuddy();

    byteBuddy
        .redefine(ByteBuddyFoo.class)
        .method(ElementMatchers.named("anotherMethod").or(ElementMatchers.named("toString")))
        .intercept(MethodDelegation.to(MethodInterceptor.class))
        .make()
        .load(ByteBuddyFoo.class.getClassLoader(), ClassReloadingStrategy.fromInstalledAgent());

我的简单 POJO 是:

public class ByteBuddyFoo {
  @Override
  public String toString() {
    return "ByteBuddyFoo{}";
  }

  public void anotherMethod(final String input) {
    System.out.println("input from anotherMethod: " + input);
  }
}

当我的拦截器中没有 @Origin Method 方法时,它可以很好地拦截调用;但是,一旦我将其添加回来,它就会抛出上述异常。

我希望能够拦截我想要的任何方法调用,并有权访问方法、其参数和结果。最初,我是通过 AspectJ 这样做的;但是,我希望能够在运行时执行此操作,这样我的代码就不需要修改。

java byte-buddy
1个回答
1
投票

您需要禁用

Method
实例的缓存,如下所示:

@Origin(cache = false)

为此,但这使得通话相当昂贵。如果您只想对方法实例进行字符串化,请使用

@Origin String method
,它无论如何都不需要缓存

重新定义时,应该使用

.disableClassFormatChanges()

© www.soinside.com 2019 - 2024. All rights reserved.