blob: ab83847bfcedb521d5e97282339cbfd679096336 [file] [log] [blame]
//
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
//
= 使用 CDI 中的事件
:jbake-type: tutorial
:jbake-tags: tutorials
:jbake-status: published
:icons: font
:syntax: true
:source-highlighter: pygments
:toc: left
:toc-title:
:description: 使用 CDI 中的事件 - Apache NetBeans
:keywords: Apache NetBeans, Tutorials, 使用 CDI 中的事件
_撰稿人:Andy Gibson_
== 上下文和依赖关系注入
1. link:cdi-intro.html[+CDI JSF 2.0 入门指南+]
2. link:cdi-inject.html[+使用 CDI 中的注入和限定符+]
3. link:cdi-validate.html[+应用 @Alternative Bean 和生命周期标注+]
4. *使用 CDI 中的事件*
* <<event,使用事件>>
* <<scopes,处理范围>>
* <<seealso,另请参见>>
link:http://jcp.org/en/jsr/detail?id=299[+JSR-299+] 指定的上下文和依赖关系注入 (CDI) 是 Java EE 6 的一个组成部分,提供了一个体系结构,以允许 Java EE 组件(例如 Servlet、企业 Bean 和 JavaBeans)在具有明确定义范围的应用程序生命周期内存在。此外,CDI 服务允许 Java EE 组件(例如 EJB 会话 Bean 和 JavaServer Faces (JSF) 受管 Bean)注入并通过触发和观察事件以松散耦合的方式进行交互。
本教程基于 Andy Gibson 发布的博客,标题为:link:http://www.andygibson.net/blog/index.php/2010/01/11/getting-started-with-jsf-2-0-and-cdi-part-3/[+CDI 入门指南,第 3 部分 - 事件+]。它演示了如何利用 Java EE _事件_的概念,即用一种维护生成器和观察器之间的分离代码的方式,来产生和订阅(即_观察_)在应用程序中发生的事件。使用 `javax.enterprise.event.Event` 类创建事件,并使用 CDI 的 `@Observes` 标注订阅事件。
NetBeans IDE 为上下文和依赖关系注入提供了内置支持,包括在项目创建时构建 `beans.xml` CDI 配置文件的选项,为标注提供的编辑器和导航支持,以及用于创建常用 CDI 工件的各种向导。
要学完本教程,您需要具备以下软件和资源。
|===
|软件或资源 |要求的版本
|link:https://netbeans.org/downloads/index.html[+NetBeans IDE+] |7.2、7.3、7.4、8.0、Java EE 版本
|link:http://www.oracle.com/technetwork/java/javase/downloads/index.html[+Java 开发工具包 (JDK)+] |版本 7 或 8
|link:http://glassfish.dev.java.net/[+GlassFish Server+] |Open Source Edition 3.x 或 4.x
|link:https://netbeans.org/projects/samples/downloads/download/Samples%252FJavaEE%252FcdiDemo3.zip[+cdiDemo3.zip+] |N/A
|===
[NOTE]
====
* NetBeans IDE Java EE 包中还含 GlassFish Server Open Source Edition,后者是与 Java EE 兼容的容器。
* 可以下载本教程的解决方案样例项目:link:https://netbeans.org/projects/samples/downloads/download/Samples%252FJavaEE%252FcdiDemoComplete.zip[+cdiDemoComplete.zip+]
====
[[event]]
== 使用事件
在上一教程link:cdi-validate.html[+应用 @Alternative Bean 和生命周期标注+]中,我们有一个应用程序,可以获取各项的列表、对各项进行验证并在发现无效项时采取特定操作。假定将来我们要展开系统以在发现无效项时处理发生的所有情况。这可能包括发送电子邮件、更改其他数据(例如取消订单)、或在文件或数据库表中存储拒绝列表。要完全分离实现,我们可以使用 Java EE 中的_事件_。事件由事件_生成器_发出,并由事件_观察器_订阅。与大多数 CDI 一样,事件的创建和订阅是类型安全的,且允许限定符决定观察器将观察哪些事件。
使用我们在本系列的上一教程中构建的应用程序,不需要进行任何更改即可实现此功能。我们还提供了 `ItemErrorHandler`(在link:cdi-validate.html[+上一教程+]中创建)的另一个实现,可以在每次处理项时发出事件。我们将此类命名为 `EventItemHandler`,将其注入 `ItemProcessor`,并使用 `Notify` 限定符选择它进行注入。
image::images/cdi-diagram-events.png[title="在应用程序中使用 CDI 注入对类进行松散耦合"]
1. 首先,从 `cdiDemo3.zip` 文件提取样例启动项目(请参见上面的<<requiredSoftware,所需资源列表>>)。在 IDE 中打开项目,方法是选择 "File"(文件)> "Open Project"(打开项目)(Ctrl-Shift-O 组合键;在 Mac 上为 ⌘-Shift-O 组合键),然后从计算机上的相应位置选择该项目。
2. 创建一个类,名为 `EventItemHandler`。单击 "New File"(新建文件)(image:images/new-file-btn.png[]) 按钮,或者按 Ctrl-N 组合键(在 Mac 上为 ⌘-N 组合键)以打开文件向导。
3. 选择 "Java" 类别,然后选择 "Java Class"Java 类)。单击 "Next"(下一步)。
4. 键入 *EventItemHandler* 作为类名,然后输入 *exercise4* 作为包。
5. 单击 "Finish"(完成)。新类和包生成,并在编辑器中打开此类。
6. 按如下方式实现 *EventItemHandler*。
[source,java]
----
public class EventItemHandler *implements ItemErrorHandler* {
*@Inject
private Event<Item> itemEvent;
@Override
public void handleItem(Item item) {
System.out.println("Firing Event");
itemEvent.fire(item);
}*
}
----
我们会注入 `Event` 实例,其中事件有效负载为 `Item`。事件有效负载是指从事件生成器传递到事件观察器的状态数据,在本例中传递的是已拒绝的项。当处理无效项时,我们会引发事件并传入接收的无效项。此基于事件的项处理程序的插入方式与其他任何项处理程序的方式相同,因此我们可以在任何需要的时候将其换入换出,还可以在测试过程中进行替换。
. 修复所有导入。在编辑器中右键单击并选择 "Fix Imports"(修复导入),或者按 Ctrl-Shift-I 组合键(在 Mac 上按 ⌘-Shift-I 组合键)。请务必选择 `javax.enterprise.event.Event` 作为 `Event` 类的全限定名称。
image::images/fix-all-imports.png[title="右键单击编辑器,然后选择 "Fix Imports"(修复导入)以调出 "Fix Imports"(修复导入)对话框"]
[tips]#在 `Event` 上按 Ctrl-空格键以查看该类的 Javadoc 定义。另外,还定义了上面使用的 `fire()` 方法。#
image::images/event-javadoc.png[title="按 Ctrl-空格键可查看有关 API 中类的 Javadoc 文档"]
. 创建一个限定符,名为 `Notify`。(link:cdi-inject.html[+使用 CDI 中的注入和限定符+]对限定符进行了介绍。)
. 单击 "New File"(新建文件)(image:images/new-file-btn.png[]) 按钮,或者按 Ctrl-N 组合键(在 Mac 上为 ⌘-N 组合键)以打开文件向导。
. 选择 "Context and Dependency Injection"(上下文和依赖关系注入)类别,然后选择 "Qualifier Type"(限定符类型)。单击 "Next"(下一步)。
. 输入 *Notify* 作为类名,然后输入 *exercise4* 作为包。
. 单击 "Finish"(完成)。新的 `Notify` 限定符在编辑器中打开。
[source,java]
----
@Qualifier
@Retention(RUNTIME)
@Target({METHOD, FIELD, PARAMETER, TYPE})
public @interface Notify {
}
----
. `@Notify` 标注添加到 `EventItemHandler`
[source,java]
----
*@Notify*
public class EventItemHandler implements ItemErrorHandler {
...
}
----
我们创建了一个 `@Notify` 限定符标注为注入标识此错误处理程序,并可以通过将其添加到注入点以在我们的 `ItemProcessor` 中使用。
. `exercise2.ItemProcessor` 中,将 `@Notify` 标注添加到 `EventItemHandler` 的注入点。
[source,java]
----
@Named
@RequestScoped
public class ItemProcessor {
@Inject @Demo
private ItemDao itemDao;
@Inject
private ItemValidator itemValidator;
@Inject *@Notify*
private ItemErrorHandler itemErrorHandler;
public void execute() {
List<Item> items = itemDao.fetchItems();
for (Item item : items) {
if (!itemValidator.isValid(item)) {
itemErrorHandler.handleItem(item);
}
}
}
}
----
(使用编辑器的提示为 `exercise4.Notify` 添加 import 语句。)
. 单击 "Run Project"(运行项目)(image:images/run-project-btn.png[]) 按钮以运行项目。
. 在浏览器中,单击 `Execute` 按钮,然后返回至 IDE,并在 "Output"(输出)窗口(Ctrl-4 组合键;在 Mac 上为 ⌘-4 组合键)中查看服务器日志。因为已构建的应用程序目前使用 `DefaultItemDao` 设置四个 `Item`,然后在 `Item` 上应用 `RelaxedItemValidator`,您会看到 `itemErrorHandler` 引发两次。
image::images/output-window.png[title="查看显示在 "Output"(输出)窗口中的 GlassFish Server 日志"]
目前我们还无法观察事件。不过,可以通过使用 `@Observes` 标注创建 _observer_ 方法来解决此问题。只需执行此操作即可观察事件。为了进行演示,我们可以通过添加调用其 `handleItem()` 方法的观察器方法来修改 `FileErrorReporter`(已在link:cdi-validate.html[+上一教程+]中创建),以响应引发事件。
. 要使我们的 `FileErrorReporter` 响应事件,请向类中添加以下方法。
[source,java]
----
public class FileErrorReporter implements ItemErrorHandler {
*public void eventFired(@Observes Item item) {
handleItem(item);
}*
...
}
----
(使用编辑器的提示为 `javax.enterprise.event.Observes` 添加 import 语句。)
. 再次运行项目(F6;在 Mac 上为 fn-F6),单击 `Execute` 按钮,然后返回至 IDE 并检查 "Output"(输出)窗口中的服务器日志。
image::images/output-window2.png[title="查看显示在 "Output"(输出)窗口中的 GlassFish Server 日志"]
可以看到,事件与之前一样会在无效对象上引发,但是现在每次引发事件时都将保存项信息。您可能还会注意到,可以观察生命周期事件,因为为每个引发事件创建和关闭了 `FileErrorReporter` Bean。(有关诸如 `@PostConstruct` `@PreDestroy` 等生命周期标注的讨论,请参见link:cdi-validate.html[+应用 @Alternative Bean 和生命周期标注+]。)
如以上步骤所示,`@Observes` 标注提供了一种简单的方式来观察事件。
还可以使用限定符标注事件和观察器,使观察器仅能够观察一个项的特定事件。有关演示,请参见 link:http://www.andygibson.net/blog/index.php/2010/01/11/getting-started-with-jsf-2-0-and-cdi-part-3/[+CDI 入门指南,第 3 部分 - 事件+]。
[[scopes]]
== 处理范围
就应用程序的现状而言,每次发出事件时都会创建一个 `FileErrorReporter` Bean。在这种情况下,我们不希望每次都创建新 Bean,因为我们不希望打开和关闭每个项的文件。但是仍然希望在启动进程时打开文件,然后在进程结束以后关闭文件。因此,需要考虑 `FileErrorReporter` Bean _范围_
目前,`FileErrorReporter` Bean 没有定义范围。当没有定义范围时,CDI 使用默认的伪依赖型范围。实际上,这意味着在非常短的时间范围内创建和销毁 Bean,通常在方法调用期间进行。在当前方案中,Bean 是在引发事件过程中创建和销毁的。要解决此问题,我们可以通过手动添加范围标注来延长 Bean 的范围。我们会将此 Bean 标注为 `@RequestScoped`,以便在引发第一个事件过程中创建 Bean 时,此 Bean 在请求过程中一直存在。这还意味着,对于限定注入此 Bean 的任何注入点,将注入同一 Bean 实例。
1. `FileErrorReporter` 中为 `javax.enterprise.context.RequestScoped` 添加 `@RequestScope` 标注和相应的 import 语句。
[source,java]
----
*import javax.enterprise.context.RequestScoped;*
...
*@RequestScoped*
public class FileErrorReporter implements ItemErrorHandler { ... }
----
[tips]#键入时按 Ctrl-空格键以调用编辑器的代码完成支持。通过代码完成选择项时,所有关联 import 语句都会自动添加到类中。#
image::images/code-completion.png[title="在键入时按 Ctrl-空格键可调用代码完成建议"]
. 再次运行项目(F6;在 Mac 上为 fn-F6),单击 `Execute` 按钮,然后返回至 IDE 并检查 "Output"(输出)窗口中的服务器日志。
image::images/output-window3.png[title="查看显示在 "Output"(输出)窗口中的 GlassFish Server 日志"]
请注意,仅当引发第一个事件时创建 `FileErrorReporter` Bean,并在引发最后一个事件以后将其关闭。
[source,java]
----
INFO: Firing Event
*INFO: Creating file error reporter*
INFO: Saving exercise2.Item@48ce88f6 [Value=34, Limit=7] to file
INFO: Firing Event
INFO: Saving exercise2.Item@3cae5788 [Value=89, Limit=32] to file
*INFO: Closing file error reporter*
----
事件是以模块化方式分离系统各部分的极好方法,因为事件观察器和生成器互相之间并不了解,它们也不需要进行任何配置来了解彼此。可以添加订阅事件生成器不知道观察器事件的代码片段。(如果不使用事件,则通常需要手动让事件生成器调用观察器。)例如,如果有人更新了订单状态,则可以添加一个事件,以电子邮件的形式发送给销售代表,或者如果技术支持问题已存在超过一个星期,则通知客户经理。此类规则可以在没有事件的情况下实现,但事件可以简化分离业务逻辑操作。此外,不存在编译时或构建时依赖关系。您只需要将模块添加到应用程序,这些模块便会自动开始观察和创建事件。
link:/about/contact_form.html?to=3&subject=Feedback:%20Working%20with%20Events%20in%20CDI[+发送有关此教程的反馈意见+]
[[seealso]]
== 另请参见
有关 CDI Java EE 的详细信息,请参见以下资源。
=== NetBeans 资源
* link:cdi-intro.html[+上下文和依赖关系注入以及 JSF 2.0 入门指南+]
* link:cdi-inject.html[+使用 CDI 中的注入和限定符+]
* link:cdi-validate.html[+应用 @Alternative Bean 和生命周期标注+]
* link:javaee-gettingstarted.html[+Java EE 应用程序入门指南+]
* link:../web/jsf20-intro.html[+JavaServer Faces 2.0 简介+]
=== 外部资源
* link:http://blogs.oracle.com/enterprisetechtips/entry/using_cdi_and_dependency_injection[+企业技术提示:在 JSF 2.0 应用程序中使用面向 Java 的 CDI 和依赖关系注入+]
* link:http://download.oracle.com/javaee/6/tutorial/doc/gjbnr.html[+Java EE 6 教程第五部分:面向 Java EE 平台的上下文和依赖关系注入+]
* link:http://jcp.org/en/jsr/detail?id=299[+JSR 299:上下文和依赖关系注入规范+]
* link:http://jcp.org/en/jsr/detail?id=316[+JSR 316:Java Platform、Enterprise Edition 6 规范+]