在上一篇文章中 ,我们学习了基于Observer / Event Listener和Mediator模式的基于事件的通信。 由于它们的缺点,我想展示基于事件的通信的更有效方法。 我们将从Google Guava EventBus开始,以CDI (Java EE平台的上下文和依赖注入)结束。

番石榴EventBus

Google Guava库具有有用的package eventbus 。 EventBus类允许组件之间进行发布-订阅式通信,而无需组件之间进行显式注册。 因为我们开发Web应用程序,所以我们应该将此类的实例封装在有作用域的bean中。

让我们编写EventBusProvider bean。

public class EventBusProvider implements Serializable {private EventBus eventBus = new EventBus("scopedEventBus");public static EventBus getEventBus() {// access EventBusProvider beanELContext elContext = FacesContext.getCurrentInstance().getELContext();EventBusProvider eventBusProvider =(EventBusProvider) elContext.getELResolver().getValue(elContext, null, "eventBusProvider");return eventBusProvider.eventBus;}
}

我仅以一个示例来演示Guava EventBus的所有主要功能。 让我们编写以下事件层次结构:

public class SettingsChangeEvent {}public class LocaleChangeEvent extends SettingsChangeEvent {public LocaleChangeEvent(Object newLocale) {...}
}public class TimeZoneChangeEvent extends SettingsChangeEvent {public TimeZoneChangeEvent(Object newTimeZone) {...}
}
下一步很简单。 要接收事件,对象(bean)应公开一个公共方法,该方法以@Subscribe批注进行批注,该方法接受具有所需事件类型的单个参数。 该对象需要将自身传递给EventBus实例的register()方法。 让我们创建两个bean:
public MyBean1 implements Serializable {@PostConstructpublic void initialize() throws Exception {EventBusProvider.getEventBus().register(this);}@Subscribepublic void handleLocaleChange(LocaleChangeEvent event) {// do something}@Subscribepublic void handleTimeZoneChange(TimeZoneChangeEvent event) {// do something}
}public MyBean2 implements Serializable {@PostConstructpublic void initialize() throws Exception {EventBusProvider.getEventBus().register(this);}@Subscribepublic void handleSettingsChange(SettingsChangeEvent event) {// do something}
}

要发布事件,只需将事件对象提供给EventBus实例的post()方法。 EventBus实例将确定事件的类型并将其路由到所有已注册的侦听器。

public class UserSettingsForm implements Serializable {private boolean changed;public void localeChangeListener(ValueChangeEvent e) {changed = true;        // notify subscribersEventBusProvider.getEventBus().post(new LocaleChangeEvent(e.getNewValue()));}public void timeZoneChangeListener(ValueChangeEvent e) {changed = true;        // notify subscribersEventBusProvider.getEventBus().post(new TimeZoneChangeEvent(e.getNewValue()));}public String saveUserSettings() {...if (changed) {// notify subscribersEventBusProvider.getEventBus().post(new SettingsChangeEvent());return "home";}}
}
Guava EventBus允许创建对许多不同事件做出反应的任何侦听器,只需使用@Subscribe注释许多方法即可。 侦听器可以利用现有事件层次结构。 因此,如果侦听器A正在等待事件A,并且事件A具有名为B的子类,则此侦听器将接收两种类型的事件:A和B。在我们的示例中,我们发布了三个事件:SettingsChangeEvent,LocaleChangeEvent和TimeZoneChangeEvent。 MyBean1中的handleLocaleChange()方法将仅接收LocaleChangeEvent。 方法handleTimeZoneChange()将仅接收TimeZoneChangeEvent。 但是,请查看MyBean2中的handleSettingsChange()方法。 它将接收所有三个事件!
如您所见,仍然需要手动注册(EventBusProvider.getEventBus()。register(this)),并且在上一篇文章中提到的作用域bean的问题仍然存在。 我们应该注意EventBusProvider的作用域和发布/订阅者bean的作用域。 但是,正如您可能还会看到的,与Mediator模式相比,我们有了一些改进:不需要特殊的接口,没有固定定义订户的方法名,也可以进行多侦听器,不花精力管理注册的实例,等等。但并非最不重要的是-异步AsyncEventBus和对DeadEvent的订阅(用于侦听没有侦听器调度的任何事件-便于调试)。 请按照本指南将现有的基于EventListener的系统转换为基于EventBus的系统。
CDI(上下文和依赖注入)
每个符合JEE 6的应用服务器都支持CDI(JSR-299规范)。 它定义了一组补充服务,可帮助改善应用程序代码的结构。 CDI的最著名的实现是OpenWebBeans和JBoss Weld 。 CDI中的事件允许bean完全不依赖地进行交互。 事件生产者引发事件,这些事件由容器传递给事件观察者。 这个基本架构听起来像熟悉的Observer / Observable模式,但是有很多好处。
  • 事件生产者和事件观察者彼此分离。
  • 观察者可以指定“选择器”的组合来缩小他们将接收的事件通知的范围。
  • 可以立即或延迟通知观察者,直到当前事务结束为止。
  • 使用条件观察者方法进行作用域定义时不会感到头痛(还记得作用域bean和Mediator / EventBus的问题吗?)。
条件观察者方法仅在声明观察者方法的bean范围当前处于活动状态时才允许获取已经存在的bean实例,而无需创建新的bean实例。 如果观察者方法不是有条件的,则将始终创建相应的bean。 您很灵活!
在我看来,CDI事件机制是进行基于事件的交流的最佳方法。 这个问题很复杂。 让我们只显示基本功能。 观察者方法是带有参数@Observes的bean的方法。
public MyBean implements Serializable {public void onLocaleChangeEvent(@Observes Locale locale) {...}
}

如果观察者方法仅对限定的事件感兴趣,则事件参数也可以指定限定符-这些事件具有限定符。

public void onLocaleChangeEvent(@Observes @Updated Locale locale) {...
}

事件限定符只是使用@Qualifier定义的普通限定符。 这是一个例子:

@Qualifier
@Target({FIELD, PARAMETER})
@Retention(RUNTIME)
public @interface Updated {}

事件生产者使用参数化Event接口的实例触发事件。 该接口的实例通过注入获得。 生产者通过调用Event接口的fire()方法并传递事件对象来引发事件。

public class UserSettingsForm implements Serializable {@Inject @Any Event<Locale> localeEvent;public void localeChangeListener(ValueChangeEvent e) {// notify all observerslocaleEvent.fire((Locale)e.getNewValue());}
}
容器调用所有观察者方法,并将事件对象作为事件参数的值传递。 如果任何观察者方法引发异常,则容器将停止调用观察者方法,并且该异常将由fire()方法重新抛出。 上面的@Any注释充当所有限定符的别名。 您会看到,无需手动注册观察员。 简单? 在注入点指定其他限定词也很简单:
// this will raise events to observers having parameter @Observes @Updated Locale
@Inject @Updated Event<Locale> localeEvent;
您还可以具有多个事件限定符。 该事件将传递给每个具有事件参数的观察者方法,该事件参数可以分配事件对象,并且除了与事件注入点指定的事件限定符匹配的事件限定符之外,没有任何事件限定符。 观察者方法可能具有其他参数,这些参数是注入点。 例:
public void onLocaleChangeEvent(@Observes @Updated Locale locale, User user) {...
}

动态指定限定符是什么? CDI允许通过AnnotationLiteral获得适当的限定符实例。 这样,我们可以将限定符传递给Event的select()方法。 例:

public class DocumentController implements Serializable {Document document;@Inject @Updated @Deleted Event<Document> documentEvent;public void updateDocument() {...// notify observers with @Updated annotationdocumentEvent.select(new AnnotationLiteral<Updated>(){}).fire(document);}public void deleteDocument() {...// notify observers with @Deleted annotationdocumentEvent.select(new AnnotationLiteral<Deleted>(){}).fire(document);}
}
让我们谈谈“条件观察者方法”。 默认情况下,如果当前上下文中没有观察者实例,则容器将实例化观察者以向其传递事件。 这种行为并不总是令人满意的。 我们可能只想将事件传递给当前上下文中已经存在的观察者实例。 通过在@Observes批注中添加receive = IF_EXISTS来指定条件观察者。
public void onLocaleChangeEvent(@Observes(receive = IF_EXISTS) @Updated Locale locale) {...
}
在此处阅读有关范围和上下文的更多信息。 在这篇简短的文章中,我们不能再谈论更多功能,例如“具有成员的事件限定符”和“事务观察者”。 我想鼓励大家开始学习CDI。 玩得开心!
参考: JSF中基于事件的通信。 新学派的方法 。 来自我们的JCG合作伙伴 Oleg Varaksin,来自“软件开发思想”博客。

翻译自: https://www.javacodegeeks.com/2012/07/jsf-event-based-communication-new.html

JSF基于事件的交流:新派方法相关推荐

  1. 基于jsf的项目_JSF基于事件的交流:新派方法

    基于jsf的项目 在上一篇文章中 ,我们学习了基于Observer / Event Listener和Mediator模式的基于事件的通信. 由于它们的缺点,我想展示基于事件的通信的更有效方法. 我们 ...

  2. JSF基于事件的沟通:过时的方法

    用JSF编写的Web应用程序由相互交互的bean组成. 在开发Web应用程序时,bean之间的通信是主要的设计模式之一. 有时,一个bean需要向其他bean发送事件,以通知它们某些更改或其他任何更改 ...

  3. 基于jsf的项目_JSF基于事件的沟通:过时的方法

    基于jsf的项目 用JSF编写的Web应用程序由相互交互的bean组成. 在开发Web应用程序时,bean之间的通信是主要的设计模式之一. 有时,一个bean需要将事件发送给其他bean,以通知它们某 ...

  4. 基于HLW8110的交流测量原理及校准方法

    基于HLW8110的交流测量原理及校准方法                                                                               ...

  5. 基于HLW8112的交流测量原理及校准方法

    基于HLW8112的交流测量原理及校准方法                                                                               ...

  6. 激光雷达角点检测 c语言,一种基于事件帧的角点检测方法与流程

    本发明属于图像处理领域,用于解决基于事件相机的SLAM项目的角点检测. 背景技术: 在过去几十年里,由于计算机视觉算法的研究与发展,人们对机器人感知的兴趣也日益增加.这种传统相机能够捕获相机周围环境的 ...

  7. 港科大开源 | 基于事件的双目视觉里程计

    点击上方"3D视觉工坊",选择"星标" 干货第一时间送达 1 摘要 本文提出了一种双目事件相机的视觉里程计方法.我们的系统遵循并行跟踪和建图的方法,建图模块以概 ...

  8. [译] 基于事件流构建的服务

    [译] 基于事件流构建的服务 摘要:本文属于原创,欢迎转载,转载请保留出处:https://github.com/jasonGeng88/blog 原文:https://www.confluent.i ...

  9. 最新综述 | 基于深度学习的SLAM方法:面向空间机器智能时代

    最新综述 | 基于深度学习的SLAM方法:面向空间机器智能时代 A Survey on Deep Learning for Localization and Mapping Towards the A ...

最新文章

  1. OpenAI联合创始人:AI的极限?我真的不知道!
  2. Dell R720上的系统安装问题的解决办法(关于RAID建立磁盘阵列的技术)
  3. 马云不会因支付宝事件走下神坛
  4. 外贸网络推广浅谈蜘蛛抓取频次的原则跟哪些有关?
  5. python怎么学好-怎么短时间学好python
  6. spring boot使用外置tomcat部署需要排除的依赖
  7. Juint整合Log4j
  8. 每天一道LeetCode-----回文链表
  9. 使用dubbo后尽量不用要@Reference可能引起冲突
  10. Python zmq 讲解
  11. linux中配置vsftpd
  12. 酷派D530刷机指引之民间ROM
  13. 机器学习中数据标准化相关方法
  14. 引擎国产化,VGS引领三维引擎新时空
  15. TwinCAT 3 使用XML-server
  16. 用VBA下载google图片
  17. 微信开放平台认证后怎么实现扫码登录功能
  18. dll 不是 PML.NET callable问题解决办法
  19. 机器学习Machine Learning
  20. 在线JSON转TSV工具

热门文章

  1. 局部描述符表LDT的作用+定义+初始化+跳转相关
  2. ConcurrentHashMap--自用,非教学
  3. java程序设置jvm_Java程序员应在2018年学习的3种JVM语言
  4. java agent_如何脚踏实地构建Java Agent
  5. 摆脱冷气_摆脱匿名类
  6. java minor gc_Java Minor发布计划再次进行了调整
  7. neo4j cypher_Neo4j:Cypher –避免热切
  8. 客户端渲染换为服务器端渲染_服务器与客户端渲染(AngularJS与服务器端MVC)
  9. Eclipse GlassFish 5.1就在这里!
  10. play框架配置 拦截器_如何使用Play框架为https配置SSL证书