当前位置:  开发笔记 > 编程语言 > 正文

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

如何解决《如何在Java中实现包装装饰器?》经验,为你挑选了1个好方法。

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

我不能修改对象的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 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子类/包装的对象在哪里.在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.



1> Rafael Winte..:

您可能想要使用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为您执行演员表.


作为一个性能注释,在我的单元测试中,我对方法(添加和代理方法)进行了1.000.000.000调用,反射我有大约10秒的执行时间,你的解决方案我有大约9毫秒的执行时间,做得好!
推荐阅读
Chloemw
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有