java 中装饰器_如何在Java中实现包装装饰器?

问题是创建现有对象的动态增强版本.

我无法修改对象的类.相反,我必须:

>继承它

>将现有对象包装在新类中

>将所有原始方法调用委托给包装对象

>实现由另一个接口定义的所有方法

添加到现有对象的接口是:

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) proxyType;

EnhancedNode enhancedNode = (EnhancedNode) enhancedClass.newInstance();

enhancedNode.setNode(node);

其中Node是子类/ wrap的对象. 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.

解决方法:

您可能希望使用Pipe而不是反射API:

public class NodeInterceptor {

@RuntimeType

public static Object intercept(@Pipe Function pipe,

@FieldValue("node") Node proxy) throws Exception {

return proxy != null

? pipe.apply(proxy);

: null;

}

}

要使用管道,首先需要安装它.如果您有Java 8可用,则可以使用java.util.Function.否则,只需定义一些类型:

interface Function { S apply(T t); }

你自己.类型和方法的名称无关紧要.安装类型:

MethodDelegation.to(NodeInterceptor.class)

.appendParameterBinder(Pipe.Binder.install(Function.class));

您是否确定反射部分是应用程序性能问题的关键点?您是否正确缓存生成的类并且缓存是否有效?反射API比它的声誉更快,特别是因为使用Byte Buddy倾向于暗示单态呼叫站点.

最后,一些一般反馈.你在打电话

.implement(EnhancedNode.class).intercept(FieldAccessor.ofBeanProperty())

多次.这没有效果.此外,method.getDeclaringClass().cast(node)不是必需的.反射API为您执行演员表.

标签:java,decorator,bytecode,byte-buddy

来源: https://codeday.me/bug/20191006/1861077.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值