在本教程中,我们将向您展示如何避免CDI bean中的依赖关系消除歧义。 在CDI中,我们可以为应用程序中不同客户端的接口的多个实现实现依赖项注入。 依赖关系消除歧义的问题是客户端如何在不同的实现中调用特定的实现,而不会发生任何错误。
为了了解在将bean注入应用程序时如何避免依赖消除歧义,我们将创建一个简单的服务。 我们将创建服务的两个实现,然后将这两个实现注入到应用程序的servlet中。 我们将使用@Qualifiers
,如下所述。
我们首选的开发环境是Eclipse 。 我们正在使用Eclipse Juno(4.2)版本以及Maven Integration插件版本3.1.0。 您可以从Eclipse的这里从和Maven Eclipse插件这里 。 用于Eclipse的Maven插件的安装不在本教程的讨论范围内,因此将不予讨论。 Tomcat 7是使用的应用程序服务器。
让我们开始,
1.创建一个新的Maven项目
转到文件->项目-> Maven-> Maven项目。
在向导的“选择项目名称和位置”页面中,确保未选中 “创建简单项目(跳过原型选择)”选项,单击“下一步”以继续使用默认值。
在这里,必须添加用于创建Web应用程序的Maven原型。 单击“添加原型”并添加原型。 将“ Archetype组ID”变量设置为"org.apache.maven.archetypes"
,将“ Archetype构件ID”变量设置为"maven-archetype-webapp"
,将“ Archetype版本”设置为"1.0"
。 点击“确定”继续。
在向导的“输入工件ID”页面中,您可以定义项目的名称和主程序包。 将“ Group Id”变量设置为"com.javacodegeeks.snippets.enterprise"
,将“ Artifact Id”变量设置为"cdibeans"
。 上述选择组成主体工程包作为"com.javacodegeeks.snippets.enterprise.cdibeans"
和项目名称为"cdibeans"
。 将“ Package”变量设置为"war"
,以便创建一个war文件以部署到tomcat服务器。 点击“完成”退出向导并创建您的项目。
Maven项目结构如下所示:
- 它包含以下文件夹:
- / src / main / java文件夹,其中包含应用程序动态内容的源文件,
- / src / test / java文件夹包含用于单元测试的所有源文件,
- / src / main / resources文件夹包含配置文件,
- / target文件夹包含已编译和打包的可交付成果,
- / src / main / resources / webapp / WEB-INF文件夹包含Web应用程序的部署描述符,
- pom.xml是项目对象模型(POM)文件。 包含所有项目相关配置的单个文件。
2.添加所有必要的依赖项
您可以通过在POM编辑器的“ Pom.xml”页面上对其进行编辑,来在Maven的pom.xml
文件中添加依赖项,如下所示:
pom.xml:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.javacodegeeks.snippets.enterprise.cdi</groupId>
<artifactId>cdibeans</artifactId>
<packaging>war</packaging>
<version>0.0.1-SNAPSHOT</version>
<name>cdibeans Maven Webapp</name>
<url>http://maven.apache.org</url>
<dependencies>
<dependency>
<groupId>org.jboss.weld.servlet</groupId>
<artifactId>weld-servlet</artifactId>
<version>1.1.10.Final</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>jstl</artifactId>
<version>1.2</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.0.1</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>org.glassfish</groupId>
<artifactId>javax.faces</artifactId>
<version>2.1.7</version>
</dependency>
</dependencies>
<build>
<finalName>cdibeans</finalName>
</build>
</project>
如您所见,Maven以声明方式管理库依赖关系。 创建本地存储库(默认情况下,位于{user_home} /。m2文件夹下),所有必需的库都从公共存储库下载并放置在该库中。 此外,库内的依赖关系会自动解决和处理。
3.创建一个简单的服务
我们利用一个简单的服务为使用它的应用程序创建问候消息。 GreetingCard.java
类是带有产生问候消息的方法的接口。
GreetingCard.java
package com.javacodegeeks.snippets.enterprise.cdibeans;
public interface GreetingCard {
void sayHello();
}
我们创建该服务的两个实现。 每个实现都会产生不同的消息,如下所示:
GreetingCardImpl.java
package com.javacodegeeks.snippets.enterprise.cdibeans.impl;
import com.javacodegeeks.snippets.enterprise.cdibeans.GreetingCard;
public class GreetingCardImpl implements GreetingCard {
public void sayHello() {
System.out.println("Hello!!!");
}
}
AnotherGreetingCardImpl.java
package com.javacodegeeks.snippets.enterprise.cdibeans.impl;
import com.javacodegeeks.snippets.enterprise.cdibeans.GreetingCard;
public class AnotherGreetingCardImpl implements GreetingCard {
public void sayHello() {
System.out.println("Have a nice day!!!");
}
}
4.服务的使用
为了将服务注入另一个bean,我们可以使用@Qualifier
。 CDI允许我们创建自己的Java批注,然后在应用程序的注入点中使用它,以根据Bean的GreetingType
获得正确的GreetingCard
实现。
Greetings.java
package com.javacodegeeks.snippets.enterprise.cdibeans;
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.ElementType.TYPE;
import static java.lang.annotation.ElementType.METHOD;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import javax.inject.Qualifier;
@Qualifier
@Retention(RUNTIME)
@Target({ FIELD, TYPE, METHOD })
public @interface Greetings {
GreetingType value();
}
GreetingType.java
是一个枚举,如下所示:
GreetingType.java
package com.javacodegeeks.snippets.enterprise.cdibeans;
public enum GreetingType {
HELLO, HI;
}
现在,服务实现使用注释,如下所示:
GreetingCardImpl.java
package com.javacodegeeks.snippets.enterprise.cdibeans.impl;
import com.javacodegeeks.snippets.enterprise.cdibeans.GreetingCard;
@Greetings(GreetingType.HELLO)
public class GreetingCardImpl implements GreetingCard {
public void sayHello() {
System.out.println("Hello!!!");
}
}
AnotherGreetingCardImpl.java
package com.javacodegeeks.snippets.enterprise.cdibeans.impl;
import com.javacodegeeks.snippets.enterprise.cdibeans.GreetingCard;
@Greetings(GreetingType.HI)
public class AnotherGreetingCardImpl implements GreetingCard {
public void sayHello() {
System.out.println("Have a nice day!!!");
}
}
5.将服务注入到servlet中
我们创建一个简单的servlet,并使用CDI提供的@Inject
注释注入服务的两种实现,如下所示:
GreetingServlet.java
package com.javacodegeeks.snippets.enterprise.cdibeans.servlet;
import java.io.IOException;
import java.io.PrintWriter;
import javax.inject.Inject;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.javacodegeeks.snippets.enterprise.cdibeans.GreetingCard;
import com.javacodegeeks.snippets.enterprise.cdibeans.GreetingType;
import com.javacodegeeks.snippets.enterprise.cdibeans.Greetings;
@WebServlet(name = "greetingServlet", urlPatterns = {"/sayHello"})
public class GreetingServlet extends HttpServlet {
private static final long serialVersionUID = 2280890757609124481L;
@Inject
@Greetings(GreetingType.HELLO)
private GreetingCard greetingCard;
@Inject
@Greetings(GreetingType.HI)
private GreetingCard anotherGreetingCard;
public void init() throws ServletException {
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html");
PrintWriter out = response.getWriter();
out.println("<h1>" + greetingCard.sayHello() + "</h1>");
out.println("<h1>" + anotherGreetingCard.sayHello() + "</h1>");
}
public void destroy(){
}
}
要运行该示例,我们必须使用Maven构建项目,然后将生成的war
文件放置在tomcat的webbaps
文件夹中。 然后,我们可以继续:
http://localhost/8080/cdibeans/sayHello
结果是如下所示:
请注意,如Java EE CDI Producer方法教程中所示,使用Producer
方法注入CDI bean时,依赖关系也可能会消除歧义。
这是Java EE CDI依赖关系歧义教程。 下载本教程的源代码: CDIDependencyDisambiguationExample.zip
翻译自: https://www.javacodegeeks.com/2013/06/java-ee-cdi-dependency-disambiguation-example.html