【发布时间】:2016-04-08 07:20:35
【问题描述】:
问题是创建现有对象的动态增强版本。
我无法修改对象的Class。相反,我必须:
- 子类化它
- 将现有对象包装在新的
Class中 - 将所有原始方法调用委托给包装对象
- 实现由另一个接口定义的所有方法
添加到现有对象的接口是:
public interface EnhancedNode {
Node getNode();
void setNode(Node node);
Set getRules();
void setRules(Set rules);
Map getGroups();
void setGroups(Map groups);
}
有了Byte Buddy,我设法继承并实现了我的接口。问题是对包装对象的委托。我发现这样做的唯一方法是使用太慢的反射(我的应用程序负载很重,性能很关键)。
到目前为止,我的代码是:
Class<? extends Node> proxyType = new ByteBuddy()
.subclass(node.getClass(), ConstructorStrategy.Default.IMITATE_SUPER_TYPE_PUBLIC)
.method(anyOf(finalNode.getClass().getMethods())).intercept(MethodDelegation.to(NodeInterceptor.class))
.defineField("node", Node.class, Visibility.PRIVATE)
.implement(EnhancedNode.class).intercept(FieldAccessor.ofBeanProperty())
.defineField("groups", Map.class, Visibility.PRIVATE)
.implement(EnhancedNode.class).intercept(FieldAccessor.ofBeanProperty())
.defineField("rules", Set.class, Visibility.PRIVATE)
.implement(EnhancedNode.class).intercept(FieldAccessor.ofBeanProperty())
.make()
.load(getClass().getClassLoader(), ClassLoadingStrategy.Default.WRAPPER)
.getLoaded();
enhancedClass = (Class<N>) proxyType;
EnhancedNode enhancedNode = (EnhancedNode) enhancedClass.newInstance();
enhancedNode.setNode(node);
其中Node 是要子类化/包装的对象。 NodeInterceptor 将调用的方法转发到getNode 属性。
这里是NodeInterceptor的代码:
public class NodeInterceptor {
@RuntimeType
public static Object intercept(@Origin Method method,
@This EnhancedNode proxy,
@AllArguments Object[] arguments)
throws Exception {
Node node = proxy.getNode();
Object res;
if (node != null) {
res = method.invoke(method.getDeclaringClass().cast(node), arguments);
} else {
res = null;
}
return res;
}
}
一切正常但拦截方法太慢,我打算直接使用ASM添加Node的每个方法的实现但我希望有一个更简单的方法使用Byte Buddy。
【问题讨论】:
-
我不知道 Byte-Buddy,但我看到一个新的代理类被一遍又一遍地创建的机会。是否有某种缓存?
-
顺便说一句,你如何强制给定节点的类具有公共默认构造函数?
-
你有一个接口,你添加了字段和方法......为什么你不能自己创建一个包装类没有这个动态增强?
-
@HannoBinder 该类为每个 Node 子类缓存,您是对的,这些类需要默认构造函数,这是另一个未解决的问题 ;)
-
@AdamSkywalker 我需要动态增强器,因为节点位于巨大的集合中,如果我将它们包装在不同的类中,我必须在每次客户端需要节点子集时循环遍历集合以提取它们,这是表演的问题。使用子类,我可以返回包装器集合而无需转换
标签: java decorator bytecode byte-buddy