ByteBuddy MethodDelegation在Java代理中不起作用

时间:2017-03-02 15:06:19

标签: java byte-buddy

我有一个premain(),其中所有使用某个注释注释的方法都应该委托给某个类。一般来说,我看起来像这样:

public static void premain( final String agentArguments, final Instrumentation instrumentation ) {

  CountingInterception ci = new CountingInterception();

  new AgentBuilder.Default()
    .type(ElementMatchers.isAnnotatedWith(com.codahale.metrics.annotation.Counted.class))
      .transform((builder, type, classLoader, module) ->
         builder.method(ElementMatchers.any())
                .intercept(MethodDelegation.to(ci))
      ).installOn(instrumentation);
}

使用调试器显示已处理此部分,但如果调用带注释的方法,则不会发生任何事情。

CountingInterception看起来像这样

public class CountingInterception {

  @RuntimeType
  public Object intercept(@DefaultCall final Callable<?> zuper, @Origin final Method method, @AllArguments final Object... args) throws Exception {

    String name = method.getAnnotation(Counted.class).name();
    if (name != null) {
        // do something
    }

    return zuper.call();
  }
}

感谢任何提示!

使用ByteBuddy 1.6.9

2 个答案:

答案 0 :(得分:2)

为了实现我想做的事情,我做了以下更改:

在premain:

CountingInterception ci = new CountingInterception();

new AgentBuilder.Default()
    .type(declaresMethod(isAnnotatedWith(Counted.class)))
      .transform((builder, type, classLoader, module) -> builder
        .method(isAnnotatedWith(Counted.class))
                 .intercept(MethodDelegation.to(ci).andThen(SuperMethodCall.INSTANCE))
      ).installOn(instrumentation);

并在CountingInterception中:

public void interceptor(@Origin final Method method) throws Exception {

    String name = method.getAnnotation(Counted.class).name();
    if (name != null) {
      // do something
    }

}

答案 1 :(得分:1)

我假设你正在尝试做一些与Java 8默认方法调用不同的事情。你的意思是使用调用超级方法的@SuperCall吗?

我建议你: 1.减少你的拦截器什么也不做。创建一个拦截器,将MethodDelegationSuperMethodCall链接起来。 2.注册AgentBuilder.Listener以将错误写入控制台。

我确信Byte Buddy无法绑定您的方法,因为您的拦截器只能应用于提供默认方法实现的类。