为什么我的 ByteBuddy 拦截器没有被调用?

Why isn't my ByteBuddy interceptor called?

我正在尝试使用 ByteBuddy 1.4.26 拦截对 JavaFX 的 ScenePulseListener.pulse() 方法的调用。

下面的代码似乎正确引用了这个class和方法,拦截器的方法签名似乎是正确,因为篡改会正确触发异常。

(在加载 ScenePulseListener class 之前调用 installJavaFXPulseInterceptor()。)

但是,当 ScenePulseListener.pulse() 是:我做错了什么时,永远不会调用拦截器方法?

public static class PulseInterceptor
{
    public static void interceptPulse( @SuperCall final Callable< Void >  pSuper )
        throws Exception
    {
        pSuper.call();
    }
}

public static final void installJavaFXPulseInterceptor()
{
    final TypePool            type_pool= TypePool.Default.ofClassPath();

    (new ByteBuddy())
        .rebase( type_pool.describe( "javafx.scene.Scene$ScenePulseListener" ).resolve(),
                 ForClassLoader.ofClassPath() )
        .method( named( "pulse" ) )
        .intercept( MethodDelegation.to( PulseInterceptor.class ) )
        .make()
        .load( ClassLoader.getSystemClassLoader(), ClassLoadingStrategy.Default.INJECTION );
}

您不应使用此方法,因为您无法保证在尝试注入类型时 class 尚未加载。这实际上仅在您的代码之前没有加载任何内容时才有效,即直接在调用 main 方法之后。

理想情况下,您应该为此类转换定义一个 Java 代理。

您可以通过定义 class:

来验证您的拦截器
public class Foo {
  public void pulse() {
    System.out.println("foo");
  }
}

看看你的改造是否有效。确保在转换之前不加载 class。在我的机器上执行此操作表明您的代码示例已经加载。

您确定系统 class 加载程序正在加载您的 JavaFx class 吗? Pulse.class.getClassLoader() 说什么?