基于jsf的项目

在上一篇文章中 ,我们学习了基于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允许创建对许多不同事件做出React的任何侦听器,只需使用@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的项目

基于jsf的项目_JSF基于事件的交流:新派方法相关推荐

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

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

  2. 【论文学习】基于贝叶斯分类的大学生关注热点事件微博文本分类方法研究——殷复莲 张晓宇 冯晴

    自己梳理一遍论文,作学习用. 基本信息 作者:殷复莲 张晓宇 冯晴(中国传媒大学信息工程学院) 摘要:针对互联网,尤其是微博平台中大学生用户产生的海量的文本分类问题,应用贝叶斯网络分类器对微博评论进行 ...

  3. JSF基于事件的交流:新派方法

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

  4. 基于JSF框架的在线棋牌游戏平台

    一.项目总体说明 1.1  项目总体说明 该项目是基于JSF框架的在线棋牌游戏平台,目前有五子棋和象棋两项已完成子模块,皆支持人机对战与在线匹配对战,其中象棋还支持大部分棋牌平台不支持的复盘功能.项目 ...

  5. 录制元素基于smartgwt框架项目如何开展自动化(一)

    时间紧张,先记一笔,后续优化与完善. 基于smartgwt框架项目如何开展自动化(一) 近来接手数据资源平台产品的自动化建立,该项目依附smartgwt框架,项目早期尝试应用watir.webdriv ...

  6. jap页面使用ajax动态加载列表数据,JSF - 使用AJAX调用基于DataTable中的mimetype值加载不同形式...

    对于我的一个项目,我需要根据DataTable组件中可用的mimetype值加载不同的表单.我有标题,内容 - 中间的DataTable和页脚中的数据显示.单击DataTable条目应根据mimety ...

  7. 基于深度学习的异构时序事件患者数据表示学习框架

    基于深度学习的异构时序事件患者数据表示学习框架 刘卢琛, 沈剑豪, 张铭,, 王子昌, 李浩然, 刘泽群 北京大学信息科学技术学院,北京 100871   摘要:患者数据的表示学习可以将患者历史信息综 ...

  8. 20个基于DPDK开源项目,建议收藏

    一.ANS – DPDK 原生加速网络堆栈 ANS(加速网络堆栈)是DPDK本地TCP/IP堆栈,也参考FreeBSD实现.ANS提供了一个与Intel DPDK一起使用的用户空间TCP/IP堆栈. ...

  9. 自动化运维-----项目实战: 基于Ansible的云平台自动化运维系统

    文章目录 项目实战: 基于Ansible的云平台自动化运维系统 一.项目介绍 1.项目介绍 2.项目背景 二.项目环境搭建 1.项目目录的配置 2.远程服务器虚拟环境的配置 3.MySQL数据库配置 ...

最新文章

  1. ajax文字上下滚动,ajax页面底部 滚动加载效果 实例
  2. 字符集与编码(四)——Unicode
  3. 零基础建站第一季视频分享
  4. 使用DynamoDB映射器将DynamoDB项目映射到对象
  5. exchange 删除邮件
  6. 2012021402
  7. 个人笔记------无级分类格式化
  8. STM32跑马灯实验蜂鸣器实验按键输入实验NVIC中断优先级管理
  9. 用最简单的方法生成代码雨,效果相当的哇塞。
  10. python+gdal+numpy实现影像uint16转uint8
  11. windows中office无法使用撤销键
  12. 阿里云企业邮箱:密码登录
  13. Android so 文件全部报错:Duplicate resources
  14. python中interval_Python 数值区间处理_对interval 库的快速入门详解
  15. Manjaro学习笔记
  16. tp5源码分析之网络请求
  17. solr mysql 全文搜索_全文检索Solr集成HanLP中文分词
  18. java开发台球的图片_Java模拟桌球打击处理及绘制
  19. C#delegate、event、Action、EventHandler 的使用和区别
  20. 计算机设备与环境安全案例,2-1计算机设备与环境安全.pptx

热门文章

  1. JDK8新特性之Optional
  2. 用数组存储字符C进行回文检测
  3. CSS3中的动画示例
  4. instanceof关键字
  5. MyBatis_1 简介
  6. python 随机排序_Python 如何随机打乱列表(List)排序
  7. 设计费收费标准2002修订_@设计师,2019景观园林设计收费标准,注意查收!
  8. springboot使用ImportBeanDefinitionRegistrar 动态注册bean
  9. MySQL事务管理+安全管理+MySQL数据类型
  10. java集合——集合接口+迭代器接口