6

BEFORE コンストラクターインターセプターを作成する方法を知っています。

return builder.constructor(isDeclaredBy(typeDescription))
   .intercept(MethodDelegation.to(constructorInterceptor)
   .andThen(SuperMethodCall.INSTANCE));

AFTER コンストラクターインターセプターを作成する方法を知っています。

return builder.constructor(isDeclaredBy(typeDescription))
   .intercept(SuperMethodCall.INSTANCE
   .andThen(MethodDelegation.to(constructorInterceptor)));

次のインターセプターを使用します。

public void intercept(@Origin Constructor<?> constructor) {
    System.out.println("intercepted " + constructor.getName());
}

ただし、前後のインターセプターを作成する方法がわかりません。これが私が試したことです(メソッドですでに機能しているものに基づく単純なアプローチ):

return builder.constructor(isDeclaredBy(typeDescription))
    .intercept(MethodDelegation.to(constructorInterceptor));

このメソッド デリゲートを使用すると、次のようになります。

public void intercept(@Origin Constructor<?> constructor, @SuperCall Callable<?> zuper) throws Exception {
    System.out.println("before " + constructor.getName());

    zuper.call();

    System.out.println("after " + constructor.getName());
}

このセットアップでは、次のようになります。

java.lang.ClassFormatError: Bad method name at constant pool index 23 in class file com/flow/agent/test/Foo$auxiliary$syFGNB3u

完全なスタック トレース:

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.agent.builder.AgentBuilder$InitializationStrategy$SelfInjection$Dispatcher$Split.register(AgentBuilder.java:1017)
    at net.bytebuddy.agent.builder.AgentBuilder$Default$Transformation$Simple$Resolution.apply(AgentBuilder.java:2795)
    at net.bytebuddy.agent.builder.AgentBuilder$Default$ExecutingTransformer.transform(AgentBuilder.java:3081)
    at sun.instrument.TransformerManager.transform(TransformerManager.java:188)
    at sun.instrument.InstrumentationImpl.transform(InstrumentationImpl.java:428)
    at java.lang.ClassLoader.defineClass1(Native Method)
    at java.lang.ClassLoader.defineClass(ClassLoader.java:760)
    at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142)
    at java.net.URLClassLoader.defineClass(URLClassLoader.java:467)
    at java.net.URLClassLoader.access$100(URLClassLoader.java:73)
    at java.net.URLClassLoader$1.run(URLClassLoader.java:368)
    at java.net.URLClassLoader$1.run(URLClassLoader.java:362)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:361)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:331)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
    ...
Caused by: java.lang.ClassFormatError: Bad method name at constant pool index 23 in class file com/flow/agent/test/Foo$auxiliary$syFGNB3u
    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)
4

1 に答える 1

4

Java 仮想マシンのバリデーターでは、実装されているコンストラクターから別のコンストラクターをハードコーディングして呼び出す必要があります。したがって、@SuperCallアラウンドアドバイスの実装に使用しても、残念ながら機能しません。実際のところ、@SuperCallアノテーションはコンストラクターと一緒に機能することはできません。(理想的には、Byte Buddy がこの試みをキャッチし、より読みやすい例外をスローします。これは、ライブラリの次のバージョンで修正します。)

できることは、次のようにインターセプターを定義することです。

public class Interceptor {
  public void before(@Origin Constructor<?> constructor) {
    System.out.println("before " + constructor.getName());
  }
  public void after(Origin Constructor<?> constructor) {
    System.out.println("after " + constructor.getName());
  }
}

次のようなインターセプトを使用します。

MethodDelegation.to(constructorInterceptor).filter(named("before"))
                .andThen(SuperMethodCall.INSTANCE
                .andThen(MethodDelegation.to(constructorInterceptor))
                                         .filter(named("after")))

これによりbefore、まずメソッドが呼び出され、次にスーパー コンストラクターが呼び出され、次にafterインターセプターが呼び出されます。

もちろん、値を から に転送できるようにしたい場合もありbeforeますafter。この目的のために、Byte Buddy はまだ優れた方法を提供していません。(私はまだ、JVM 自体の機能強化を利用することを望んでいます。この VM の制限は、メソッド ハンドルを使用する人々にも影響を与え、VM を使用する人々から不幸な欠点として言及されることがよくあります。)

今のところ、いつでもThreadLocalインターセプターでフィールドをbefore定義して、 after. (シングル スレッド環境ではThreadLocal、単純なフィールドを削除して使用することもできます。) これをまだターゲットにしていない理由の 1 つは、ほとんどのコンストラクター インターセプトが around-adivce を必要としないためです。より詳細な使用例を記載していただければ、さらにお役に立てるかもしれません。

于 2015-12-24T13:17:05.460 に答える