使用 Byte Buddy 转发方法调用
Forwarding method calls with Byte Buddy
我正在尝试使用 Byte Buddy 设置一个 class 来实现给定实例的所有接口并将所有调用转发给该实例,这是我目前的代码:
import static net.bytebuddy.implementation.Forwarding.to;
static Class<?> save (state) {
return stream (state.getClass ().getMethods ())
.filter (m -> isPublic (m.getModifiers ()))
.reduce ((Builder<?>) new ByteBuddy ().subclass (Object.class)
.implement (state.getClass ().getInterfaces ()),
(b, m) -> {
System.out.println ("Setting up method " + m.getName ());
return b.define (new ForLoadedMethod (m))
.intercept (to (state));
}, (x, y) -> {
throw new UnsupportedOperationException ();
}).make ().load (state.getClass ()
.getClassLoader (), INJECTION).getLoaded ();
}
public static void main (String[] args) throws Throwable {
System.out.println ("Saved: " + save (new Integer (1)).newInstance ().toString ());
}
这会导致如下异常:
Exception in thread "main" java.lang.IllegalArgumentException: Cannot forward public boolean net.bytebuddy.renamed.java.lang.Object$ByteBuddy$rwGGy3NN.equals(java.lang.Object) to class java.lang.Integer
at net.bytebuddy.implementation.Forwarding$Appender.apply(Forwarding.java:337)
at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod$WithBody.applyBody(TypeWriter.java:510)
at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod.apply(TypeWriter.java:444)
我可以在 Forwarding
下看到说明它可以工作 the intercepted method must be defined on a super type of the given delegation target
但我不确定那是什么意思。如果那意味着我应该对生成的 class 进行子class,如果我的目标是 final
class 怎么办?我还能如何创建将转发这些调用的自定义实现?
另外,反正我也来了,如何指定一个通用的接口类型来实现呢?它是来自已加载的 java.lang.reflect.Type
还是来自我想即时创建的内容?
转发实现的工作方式类似于反射 API,其中目标需要是可分配类型。由于您没有将 Integer 子类化(无论如何这是不可能的),转发委托失败。
我认为您正在寻找 MethodCall 委托,您可以在其中显式命名要调用的方法。这对你有用吗?
最后,实现泛型类型是 0.8 版计划的一项功能。我已经取得了很好的进展,希望在 2016 年 1 月内准备好发布候选版本。我还计划重新访问转发委托,signature-based 调用应该不会太难实现。
我正在尝试使用 Byte Buddy 设置一个 class 来实现给定实例的所有接口并将所有调用转发给该实例,这是我目前的代码:
import static net.bytebuddy.implementation.Forwarding.to;
static Class<?> save (state) {
return stream (state.getClass ().getMethods ())
.filter (m -> isPublic (m.getModifiers ()))
.reduce ((Builder<?>) new ByteBuddy ().subclass (Object.class)
.implement (state.getClass ().getInterfaces ()),
(b, m) -> {
System.out.println ("Setting up method " + m.getName ());
return b.define (new ForLoadedMethod (m))
.intercept (to (state));
}, (x, y) -> {
throw new UnsupportedOperationException ();
}).make ().load (state.getClass ()
.getClassLoader (), INJECTION).getLoaded ();
}
public static void main (String[] args) throws Throwable {
System.out.println ("Saved: " + save (new Integer (1)).newInstance ().toString ());
}
这会导致如下异常:
Exception in thread "main" java.lang.IllegalArgumentException: Cannot forward public boolean net.bytebuddy.renamed.java.lang.Object$ByteBuddy$rwGGy3NN.equals(java.lang.Object) to class java.lang.Integer
at net.bytebuddy.implementation.Forwarding$Appender.apply(Forwarding.java:337)
at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod$WithBody.applyBody(TypeWriter.java:510)
at net.bytebuddy.dynamic.scaffold.TypeWriter$MethodPool$Record$ForDefinedMethod.apply(TypeWriter.java:444)
我可以在 Forwarding
下看到说明它可以工作 the intercepted method must be defined on a super type of the given delegation target
但我不确定那是什么意思。如果那意味着我应该对生成的 class 进行子class,如果我的目标是 final
class 怎么办?我还能如何创建将转发这些调用的自定义实现?
另外,反正我也来了,如何指定一个通用的接口类型来实现呢?它是来自已加载的 java.lang.reflect.Type
还是来自我想即时创建的内容?
转发实现的工作方式类似于反射 API,其中目标需要是可分配类型。由于您没有将 Integer 子类化(无论如何这是不可能的),转发委托失败。
我认为您正在寻找 MethodCall 委托,您可以在其中显式命名要调用的方法。这对你有用吗?
最后,实现泛型类型是 0.8 版计划的一项功能。我已经取得了很好的进展,希望在 2016 年 1 月内准备好发布候选版本。我还计划重新访问转发委托,signature-based 调用应该不会太难实现。