Ce contenu n'est pas disponible dans la langue sélectionnée.
Chapter 7. Contexts and Dependency Injection (CDI)
7.1. Introduction to CDI Copier lienLien copié sur presse-papiers!
7.1.1. About Contexts and Dependency Injection (CDI) Copier lienLien copié sur presse-papiers!
Contexts and Dependency Injection (CDI) 1.2 is a specification designed to enable Enterprise Java Beans (EJB) 3 components to be used as Java Server Faces (JSF) managed beans. CDI unifies the two component models and enables a considerable simplification to the programming model for web-based applications in Java. CDI 1.2 release is treated as a maintenance release of 1.1. Details about CDI 1.1 can be found in JSR 346: Contexts and Dependency Injection for Java™ EE 1.1.
JBoss EAP includes Weld, which is the reference implementation of JSR-346:Contexts and Dependency Injection for Java™ EE 1.1.
Benefits of CDI
The benefits of CDI include:
- Simplifying and shrinking your code base by replacing big chunks of code with annotations.
- Flexibility, allowing you to disable and enable injections and events, use alternative beans, and inject non-CDI objects easily.
-
Optionally, allowing you to include
beans.xml
in yourMETA-INF/
orWEB-INF/
directory if you need to customize the configuration to differ from the default. The file can be empty. - Simplifying packaging and deployments and reducing the amount of XML you need to add to your deployments.
- Providing lifecycle management via contexts. You can tie injections to requests, sessions, conversations, or custom contexts.
- Providing type-safe dependency injection, which is safer and easier to debug than string-based injection.
- Decoupling interceptors from beans.
- Providing complex event notification.
7.1.2. Relationship Between Weld, Seam 2, and JavaServer Faces Copier lienLien copié sur presse-papiers!
Weld is the reference implementation of CDI, which is defined in JSR 346: Contexts and Dependency Injection for Java™ EE 1.1. Weld was inspired by Seam 2 and other dependency injection frameworks, and is included in JBoss EAP.
The goal of Seam 2 was to unify Enterprise Java Beans and JavaServer Faces managed beans.
JavaServer Faces 2.2 implements JSR-344: JavaServer™ Faces 2.2. It is an API for building server-side user interfaces.
7.2. Use CDI to Develop an Application Copier lienLien copié sur presse-papiers!
Contexts and Dependency Injection (CDI) gives you tremendous flexibility in developing applications, reusing code, adapting your code at deployment or run-time, and unit testing. JBoss EAP includes Weld, the reference implementation of CDI. These tasks show you how to use CDI in your enterprise applications.
7.2.1. Default Bean Discovery Mode Copier lienLien copié sur presse-papiers!
The default bean discovery mode for a bean archive is annotated
. Such a bean archive is said to be an implicit bean archive
.
If the bean discovery mode is annotated
then:
-
Bean classes that do not have
bean defining annotation
and are not bean classes of sessions beans are not discovered. - Producer methods that are not on a session bean and whose bean class does not have a bean defining annotation are not discovered.
- Producer fields that are not on a session bean and whose bean class does not have a bean defining annotation are not discovered.
- Disposer methods that are not on a session bean and whose bean class does not have a bean defining annotation are not discovered.
- Observer methods that are not on a session bean and whose bean class does not have a bean defining annotation are not discovered.
All examples in the CDI section are valid only when you have a discovery mode set to all
.
Bean Defining Annotations
A bean class may have a bean defining annotation
, allowing it to be placed anywhere in an application, as defined in Bean archives. A bean class with a bean defining annotation is said to be an implicit bean.
The set of bean defining annotations contains:
-
@ApplicationScoped
,@SessionScoped
,@ConversationScoped
and@RequestScoped
annotations - All other normal scope types
-
@Interceptor
and@Decorator
annotations -
All stereotype annotations, i.e. annotations annotated with
@Stereotype
-
The
@Dependent
scope annotation
If one of these annotations is declared on a bean class, then the bean class is said to have a bean defining annotation.
Example: Bean Defining Annotation
To ensure compatibility with other JSR-330 implementations, all pseudo-scope annotations, except @Dependent
, are not bean defining annotations. However, a stereotype annotation including a pseudo-scope annotation is a bean defining annotation.
7.2.2. Exclude Beans From the Scanning Process Copier lienLien copié sur presse-papiers!
Exclude filters are defined by <exclude>
elements in the beans.xml
file for the bean archive as children of the <scan>
element. By default an exclude filter is active. The exclude filter becomes inactive, if its definition contains:
-
A child element named
<if-class-available>
with aname
attribute, and the class loader for the bean archive can not load a class for that name, or -
A child element named
<if-class-not-available>
with aname
attribute, and the class loader for the bean archive can load a class for that name, or -
A child element named
<if-system-property>
with aname
attribute, and there is no system property defined for that name, or -
A child element named
<if-system-property>
with aname
attribute and a value attribute, and there is no system property defined for that name with that value.
The type is excluded from discovery, if the filter is active, and:
- The fully qualified name of the type being discovered matches the value of the name attribute of the exclude filter, or
- The package name of the type being discovered matches the value of the name attribute with a suffix ".*" of the exclude filter, or
- The package name of the type being discovered starts with the value of the name attribute with a suffix ".**" of the exclude filter
Example 7.1. Example: beans.xml
File
- 1
- The first exclude filter will exclude all classes in
com.acme.rest
package. - 2
- The second exclude filter will exclude all classes in the
com.acme.faces
package, and any subpackages, but only if JSF is not available. - 3
- The third exclude filter will exclude all classes in the
com.acme.verbose
package if the system propertyverbosity
has the valuelow
. - 4
- The fourth exclude filter will exclude all classes in the
com.acme.ejb
package, and any subpackages, if the system propertyexclude-ejbs
is set with any value and if at the same time, thejavax.enterprise.inject.Model
class is also available to the classloader.
It is safe to annotate Java EE components with @Vetoed
to prevent them being considered beans. An event is not fired for any type annotated with @Vetoed
, or in a package annotated with @Vetoed
. For more information, see @Vetoed
.
7.2.3. Use an Injection to Extend an Implementation Copier lienLien copié sur presse-papiers!
You can use an injection to add or change a feature of your existing code.
The following example adds a translation ability to an existing class, and assumes you already have a Welcome class, which has a method buildPhrase
. The buildPhrase
method takes as an argument the name of a city, and outputs a phrase like "Welcome to Boston!".
Example: Inject a Translator Bean Into the Welcome Class
The following injects a hypothetical Translator
object into the Welcome
class. The Translator
object can be an EJB stateless bean or another type of bean, which can translate sentences from one language to another. In this instance, the Translator
is used to translate the entire greeting, without modifying the original Welcome
class. The Translator
is injected before the buildPhrase
method is called.
7.3. Ambiguous or Unsatisfied Dependencies Copier lienLien copié sur presse-papiers!
Ambiguous dependencies exist when the container is unable to resolve an injection to exactly one bean.
Unsatisfied dependencies exist when the container is unable to resolve an injection to any bean at all.
The container takes the following steps to try to resolve dependencies:
- It resolves the qualifier annotations on all beans that implement the bean type of an injection point.
-
It filters out disabled beans. Disabled beans are
@Alternative
beans which are not explicitly enabled.
In the event of an ambiguous or unsatisfied dependency, the container aborts deployment and throws an exception.
To fix an ambiguous dependency, see Use a Qualifier to Resolve an Ambiguous Injection.
7.3.1. Qualifiers Copier lienLien copié sur presse-papiers!
Qualifiers are annotations used to avoid ambiguous dependencies when the container can resolve multiple beans, which fit into an injection point. A qualifier declared at an injection point provides the set of eligible beans, which declare the same Qualifier.
Qualifiers have to be declared with a retention and target as shown in the example below.
Example: Define the @Synchronous
and @Asynchronous
Qualifiers
@Qualifier @Retention(RUNTIME) @Target({TYPE, METHOD, FIELD, PARAMETER}) public @interface Synchronous {}
@Qualifier
@Retention(RUNTIME)
@Target({TYPE, METHOD, FIELD, PARAMETER})
public @interface Synchronous {}
@Qualifier @Retention(RUNTIME) @Target({TYPE, METHOD, FIELD, PARAMETER}) public @interface Asynchronous {}
@Qualifier
@Retention(RUNTIME)
@Target({TYPE, METHOD, FIELD, PARAMETER})
public @interface Asynchronous {}
Example: Use the @Synchronous
and @Asynchronous
Qualifiers
@Synchronous public class SynchronousPaymentProcessor implements PaymentProcessor { public void process(Payment payment) { ... } }
@Synchronous
public class SynchronousPaymentProcessor implements PaymentProcessor {
public void process(Payment payment) { ... }
}
@Asynchronous public class AsynchronousPaymentProcessor implements PaymentProcessor { public void process(Payment payment) { ... } }
@Asynchronous
public class AsynchronousPaymentProcessor implements PaymentProcessor {
public void process(Payment payment) { ... }
}
'@Any'
Whenever a bean or injection point does not explicitly declare a qualifier, the container assumes the qualifier @Default
. From time to time, you will need to declare an injection point without specifying a qualifier. There is a qualifier for that too. All beans have the qualifier @Any
. Therefore, by explicitly specifying @Any
at an injection point, you suppress the default qualifier, without otherwise restricting the beans that are eligible for injection.
This is especially useful if you want to iterate over all beans with a certain bean type.
Every bean has the qualifier @Any
, even if it does not explicitly declare this qualifier.
Every event also has the qualifier @Any
, even if it was raised without explicit declaration of this qualifier.
@Inject @Any Event<User> anyUserEvent;
@Inject @Any Event<User> anyUserEvent;
The @Any
qualifier allows an injection point to refer to all beans or all events of a certain bean type.
@Inject @Delegate @Any Logger logger;
@Inject @Delegate @Any Logger logger;
7.3.2. Use a Qualifier to Resolve an Ambiguous Injection Copier lienLien copié sur presse-papiers!
You can resolve an ambiguous injection using a qualifier. Read more about ambiguous injections at Ambiguous or Unsatisfied Dependencies.
The following example is ambiguous and features two implementations of Welcome
, one which translates and one which does not. The injection needs to be specified to use the translating Welcome
.
Example: Ambiguous Injection
Resolve an Ambiguous Injection with a Qualifier
To resolve the ambiguous injection, create a qualifier annotation called
@Translating
:@Qualifier @Retention(RUNTIME) @Target({TYPE,METHOD,FIELD,PARAMETERS}) public @interface Translating{}
@Qualifier @Retention(RUNTIME) @Target({TYPE,METHOD,FIELD,PARAMETERS}) public @interface Translating{}
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Annotate your translating
Welcome
with the@Translating
annotation:Copy to Clipboard Copied! Toggle word wrap Toggle overflow Request the translating
Welcome
in your injection. You must request a qualified implementation explicitly, similar to the factory method pattern. The ambiguity is resolved at the injection point.Copy to Clipboard Copied! Toggle word wrap Toggle overflow
7.4. Managed Beans Copier lienLien copié sur presse-papiers!
Java EE establishes a common definition in the Managed Beans specification. Managed Beans are defined as container-managed objects with minimal programming restrictions, otherwise known by the acronym POJO (Plain Old Java Object). They support a small set of basic services, such as resource injection, lifecycle callbacks, and interceptors. Companion specifications, such as EJB and CDI, build on this basic model.
With very few exceptions, almost every concrete Java class that has a constructor with no parameters (or a constructor designated with the annotation @Inject
) is a bean. This includes every JavaBean and every EJB session bean.
7.4.1. Types of Classes That are Beans Copier lienLien copié sur presse-papiers!
A managed bean is a Java class. The basic lifecycle and semantics of a managed bean are defined by the Managed Beans specification. You can explicitly declare a managed bean by annotating the bean class @ManagedBean
, but in CDI you do not need to. According to the specification, the CDI container treats any class that satisfies the following conditions as a managed bean:
- It is not a non-static inner class.
-
It is a concrete class, or is annotated
@Decorator
. -
It is not annotated with an EJB component-defining annotation or declared as an EJB bean class in
ejb-jar.xml
. -
It does not implement interface
javax.enterprise.inject.spi.Extension
. -
It has either a constructor with no parameters, or a constructor annotated with
@Inject
. -
It is not annotated
@Vetoed
or in a package annotated@Vetoed
.
The unrestricted set of bean types for a managed bean contains the bean class, every superclass and all interfaces it implements directly or indirectly.
If a managed bean has a public field, it must have the default scope @Dependent
.
@Vetoed
CDI 1.1 introduces a new annotation, @Vetoed
. You can prevent a bean from injection by adding this annotation:
@Vetoed public class SimpleGreeting implements Greeting { ... }
@Vetoed
public class SimpleGreeting implements Greeting {
...
}
In this code, the SimpleGreeting
bean is not considered for injection.
All beans in a package may be prevented from injection:
@Vetoed package org.sample.beans; import javax.enterprise.inject.Vetoed;
@Vetoed
package org.sample.beans;
import javax.enterprise.inject.Vetoed;
This code in package-info.java
in the org.sample.beans
package will prevent all beans inside this package from injection.
Java EE components, such as stateless EJBs or JAX-RS resource endpoints, can be marked with @Vetoed
to prevent them from being considered beans. Adding the @Vetoed
annotation to all persistent entities prevents the BeanManager
from managing an entity as a CDI Bean. When an entity is annotated @Vetoed
, no injections take place. The reasoning behind this is to prevent the BeanManager
from performing the operations that may cause the JPA provider to break.
7.4.2. Use CDI to Inject an Object Into a Bean Copier lienLien copié sur presse-papiers!
CDI is activated automatically if CDI components are detected in an application. If you wish to customize your configuration to differ from the default, you can include META-INF/beans.xml
or WEB-INF/beans.xml
to your deployment archive.
Inject Objects into Other Objects
To obtain an instance of a class, annotate the field with
@Inject
within your bean:public class TranslateController { @Inject TextTranslator textTranslator; ...
public class TranslateController { @Inject TextTranslator textTranslator; ...
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Use your injected object’s methods directly. Assume that
TextTranslator
has a methodtranslate
:// in TranslateController class public void translate() { translation = textTranslator.translate(inputText); }
// in TranslateController class public void translate() { translation = textTranslator.translate(inputText); }
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Use an injection in the constructor of a bean. You can inject objects into the constructor of a bean as an alternative to using a factory or service locator to create them:
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Use the
Instance(<T>)
interface to get instances programmatically. TheInstance
interface can return an instance ofTextTranslator
when parameterized with the bean type.@Inject Instance<TextTranslator> textTranslatorInstance; ... public void translate() { textTranslatorInstance.get().translate(inputText); }
@Inject Instance<TextTranslator> textTranslatorInstance; ... public void translate() { textTranslatorInstance.get().translate(inputText); }
Copy to Clipboard Copied! Toggle word wrap Toggle overflow
When you inject an object into a bean, all of the object’s methods and properties are available to your bean. If you inject into your bean’s constructor, instances of the injected objects are created when your bean’s constructor is called, unless the injection refers to an instance that already exists. For instance, a new instance would not be created if you inject a session-scoped bean during the lifetime of the session.
7.5. Contexts and Scopes Copier lienLien copié sur presse-papiers!
A context, in terms of CDI, is a storage area that holds instances of beans associated with a specific scope.
A scope is the link between a bean and a context. A scope/context combination may have a specific lifecycle. Several predefined scopes exist, and you can create your own. Examples of predefined scopes are @RequestScoped
, @SessionScoped
, and @ConversationScope
.
Scope | Description |
---|---|
|
The bean is bound to the lifecycle of the bean holding the reference. The default scope for an injected bean is |
| The bean is bound to the lifecycle of the application. |
| The bean is bound to the lifecycle of the request. |
| The bean is bound to the lifecycle of the session. |
| The bean is bound to the lifecycle of the conversation. The conversation scope is between the lengths of the request and the session, and is controlled by the application. |
Custom scopes | If the above contexts do not meet your needs, you can define custom scopes. |
7.6. Named Beans Copier lienLien copié sur presse-papiers!
You can name a bean by using the @Named
annotation. Naming a bean allows you to use it directly in Java Server Faces (JSF) and Expression Language (EL).
The @Named
annotation takes an optional parameter, which is the bean name. If this parameter is omitted, the bean name defaults to the class name of the bean with its first letter converted to lower-case.
7.6.1. Use Named Beans Copier lienLien copié sur presse-papiers!
Configure Bean Names Using the @Named Annotation
Use the
@Named
annotation to assign a name to a bean.Copy to Clipboard Copied! Toggle word wrap Toggle overflow In the example above, the default name would be
greeterBean
if no name had been specified.Use the named bean in a JSF view.
<h:form> <h:commandButton value="Welcome visitors" action="#{greeter.welcomeVisitors}"/> </h:form>
<h:form> <h:commandButton value="Welcome visitors" action="#{greeter.welcomeVisitors}"/> </h:form>
Copy to Clipboard Copied! Toggle word wrap Toggle overflow
7.7. Bean Lifecycle Copier lienLien copié sur presse-papiers!
This task shows you how to save a bean for the life of a request.
The default scope for an injected bean is @Dependent
. This means that the bean’s lifecycle is dependent upon the lifecycle of the bean that holds the reference. Several other scopes exist, and you can define your own scopes. For more information, see Contexts and Scopes.
Manage Bean Lifecycles
Annotate the bean with the desired scope.
Copy to Clipboard Copied! Toggle word wrap Toggle overflow When your bean is used in the JSF view, it holds state.
<h:form> <h:inputText value="#{greeter.city}"/> <h:commandButton value="Welcome visitors" action="#{greeter.welcomeVisitors}"/> </h:form>
<h:form> <h:inputText value="#{greeter.city}"/> <h:commandButton value="Welcome visitors" action="#{greeter.welcomeVisitors}"/> </h:form>
Copy to Clipboard Copied! Toggle word wrap Toggle overflow
Your bean is saved in the context relating to the scope that you specify, and lasts as long as the scope applies.
7.7.1. Use a Producer Method Copier lienLien copié sur presse-papiers!
A producer method is a method that acts as a source of bean instances. When no instance exists in the specified context, the method declaration itself describes the bean, and the container invokes the method to obtain an instance of the bean. A producer method lets the application take full control of the bean instantiation process.
This task shows how to use producer methods to produce a variety of different objects that are not beans for injection.
Example: Use a Producer Method
By using a producer method instead of an alternative, polymorphism after deployment is allowed.
The @Preferred
annotation in the example is a qualifier annotation. For more information about qualifiers, see Qualifiers.
The following injection point has the same type and qualifier annotations as the producer method, so it resolves to the producer method using the usual CDI injection rules. The producer method is called by the container to obtain an instance to service this injection point.
@Inject @Preferred PaymentStrategy paymentStrategy;
@Inject @Preferred PaymentStrategy paymentStrategy;
Example: Assign a Scope to a Producer Method
The default scope of a producer method is @Dependent
. If you assign a scope to a bean, it is bound to the appropriate context. The producer method in this example is only called once per session.
@Produces @Preferred @SessionScoped public PaymentStrategy getPaymentStrategy() { ... }
@Produces @Preferred @SessionScoped
public PaymentStrategy getPaymentStrategy() {
...
}
Example: Use an Injection Inside a Producer Method
Objects instantiated directly by an application cannot take advantage of dependency injection and do not have interceptors. However, you can use dependency injection into the producer method to obtain bean instances.
If you inject a request-scoped bean into a session-scoped producer, the producer method promotes the current request-scoped instance into session scope. This is almost certainly not the desired behavior, so use caution when you use a producer method in this way.
The scope of the producer method is not inherited from the bean that declares the producer method.
Producer methods allow you to inject non-bean objects and change your code dynamically.
7.8. Alternative Beans Copier lienLien copié sur presse-papiers!
Alternatives are beans whose implementation is specific to a particular client module or deployment scenario.
By default, @Alternative
beans are disabled. They are enabled for a specific bean archive by editing its beans.xml
file. However, this activation only applies to the beans in that archive. From CDI 1.1 onwards, the alternative can be enabled for the entire application using the @Priority
annotation.
Example: Defining Alternatives
This alternative defines an implementation of the PaymentProcessor
class using both @Synchronous
and @Asynchronous
alternatives:
Example: Enabling @Alternative
Using beans.xml
Declaring Selected Alternatives
The @Priority
annotation allows an alternative to be enabled for an entire application. An alternative may be given a priority for the application:
-
by placing the
@Priority
annotation on the bean class of a managed bean or session bean, or -
by placing the
@Priority
annotation on the bean class that declares the producer method, field or resource.
7.8.1. Override an Injection with an Alternative Copier lienLien copié sur presse-papiers!
You can use alternative beans to override existing beans. They can be thought of as a way to plug in a class which fills the same role, but functions differently. They are disabled by default.
This task shows you how to specify and enable an alternative.
Override an Injection
This task assumes that you already have a TranslatingWelcome
class in your project, but you want to override it with a "mock" TranslatingWelcome
class. This would be the case for a test deployment, where the true Translator
bean cannot be used.
Define the alternative.
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Activate the substitute implementation by adding the fully-qualified class name to your
META-INF/beans.xml
orWEB-INF/beans.xml
file.<beans> <alternatives> <class>com.acme.MockTranslatingWelcome</class> </alternatives> </beans>
<beans> <alternatives> <class>com.acme.MockTranslatingWelcome</class> </alternatives> </beans>
Copy to Clipboard Copied! Toggle word wrap Toggle overflow
The alternative implementation is now used instead of the original one.
7.9. Stereotypes Copier lienLien copié sur presse-papiers!
In many systems, use of architectural patterns produces a set of recurring bean roles. A stereotype allows you to identify such a role and declare some common metadata for beans with that role in a central place.
A stereotype encapsulates any combination of:
- Default scope
- A set of interceptor bindings
A stereotype can also specify either:
- All beans where the stereotypes are defaulted bean EL names
- All beans where the stereotypes are alternatives
A bean may declare zero, one, or multiple stereotypes. A stereotype is an @Stereotype
annotation that packages several other annotations. Stereotype annotations may be applied to a bean class, producer method, or field.
A class that inherits a scope from a stereotype may override that stereotype and specify a scope directly on the bean.
In addition, if a stereotype has a @Named
annotation, any bean it is placed on has a default bean name. The bean may override this name if the @Named
annotation is specified directly on the bean. For more information about named beans, see Named Beans.
7.9.1. Use Stereotypes Copier lienLien copié sur presse-papiers!
Without stereotypes, annotations can become cluttered. This task shows you how to use stereotypes to reduce the clutter and streamline your code.
Example: Annotation Clutter
Define and Use Stereotypes
Define the stereotype.
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Use the stereotype.
Copy to Clipboard Copied! Toggle word wrap Toggle overflow
7.10. Observer Methods Copier lienLien copié sur presse-papiers!
Observer methods receive notifications when events occur.
CDI also provides transactional observer methods, which receive event notifications during the before completion or after completion phase of the transaction in which the event was fired.
7.10.1. Fire and Observe Events Copier lienLien copié sur presse-papiers!
Example: Fire an Event
The following code shows an event being injected and used in a method.
Example: Fire an Event with a Qualifier
You can annotate your event injection with a qualifier, to make it more specific. For more information about qualifiers, see Qualifiers.
Example: Observe an Event
To observe an event, use the @Observes
annotation.
public class AccountObserver { void checkTran(@Observes Withdrawal w) { ... } }
public class AccountObserver {
void checkTran(@Observes Withdrawal w) {
...
}
}
You can use qualifiers to observe only specific types of events.
public class AccountObserver { void checkTran(@Observes @Suspicious Withdrawal w) { ... } }
public class AccountObserver {
void checkTran(@Observes @Suspicious Withdrawal w) {
...
}
}
7.10.2. Transactional Observers Copier lienLien copié sur presse-papiers!
Transactional observers receive the event notifications before or after the completion phase of the transaction in which the event was raised. Transactional observers are important in a stateful object model because state is often held for longer than a single atomic transaction.
There are five kinds of transactional observers:
-
IN_PROGRESS
: By default, observers are invoked immediately. -
AFTER_SUCCESS
: Observers are invoked after the completion phase of the transaction, but only if the transaction completes successfully. -
AFTER_FAILURE
: Observers are invoked after the completion phase of the transaction, but only if the transaction fails to complete successfully. -
AFTER_COMPLETION
: Observers are invoked after the completion phase of the transaction. -
BEFORE_COMPLETION
: Observers are invoked before the completion phase of the transaction.
The following observer method refreshes a query result set cached in the application context, but only when transactions that update the Category tree are successful:
public void refreshCategoryTree(@Observes(during = AFTER_SUCCESS) CategoryUpdateEvent event) { ... }
public void refreshCategoryTree(@Observes(during = AFTER_SUCCESS) CategoryUpdateEvent event) { ... }
Assume we have cached a JPA query result set in the application scope:
Occasionally a Product
is created or deleted. When this occurs, we need to refresh the Product
catalog. But we have to wait for the transaction to complete successfully before performing this refresh.
The bean that creates and deletes Products
triggers events:
The Catalog
can now observe the events after successful completion of the transaction:
7.11. Interceptors Copier lienLien copié sur presse-papiers!
Interceptors allow you to add functionality to the business methods of a bean without modifying the bean’s method directly. The interceptor is executed before any of the business methods of the bean. Interceptors are defined as part of the JSR 318: Enterprise JavaBeans™ 3.1 specification.
CDI enhances this functionality by allowing you to use annotations to bind interceptors to beans.
Interception points
- Business method interception: A business method interceptor applies to invocations of methods of the bean by clients of the bean.
- Lifecycle callback interception: A lifecycle callback interceptor applies to invocations of lifecycle callbacks by the container.
- Timeout method interception: A timeout method interceptor applies to invocations of the EJB timeout methods by the container.
Enabling Interceptors
By default, all interceptors are disabled. You can enable the interceptor by using the beans.xml
descriptor of a bean archive. However, this activation only applies to the beans in that archive. From CDI 1.1 onwards the interceptor can be enabled for the whole application using the @Priority
annotation.
Example: Enabling Interceptors in beans.xml
Having the XML declaration solves two problems:
- It enables us to specify an ordering for the interceptors in our system, ensuring deterministic behavior
- It lets us enable or disable interceptor classes at deployment time.
Interceptors enabled using @Priority
are called before interceptors enabled using the beans.xml
file.
Having an interceptor enabled by @Priority
and at the same time invoked by beans.xml
, leads to a non-portable behavior. This combination of enablement should therefore be avoided in order to maintain consistent behavior across different CDI implementations.
7.11.1. Use Interceptors with CDI Copier lienLien copié sur presse-papiers!
CDI can simplify your interceptor code and make it easier to apply to your business code.
Without CDI, interceptors have two problems:
- The bean must specify the interceptor implementation directly.
- Every bean in the application must specify the full set of interceptors in the correct order. This makes adding or removing interceptors on an application-wide basis time-consuming and error-prone.
Example: Interceptors Without CDI
Use Interceptors with CDI
Define the interceptor binding type:
@InterceptorBinding @Retention(RUNTIME) @Target({TYPE, METHOD}) public @interface Secure {}
@InterceptorBinding @Retention(RUNTIME) @Target({TYPE, METHOD}) public @interface Secure {}
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Mark the interceptor implementation:
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Use the interceptor in your business code:
Copy to Clipboard Copied! Toggle word wrap Toggle overflow Enable the interceptor in your deployment, by adding it to
META-INF/beans.xml
orWEB-INF/beans.xml
:Copy to Clipboard Copied! Toggle word wrap Toggle overflow
The interceptors are applied in the order listed.
7.12. Decorators Copier lienLien copié sur presse-papiers!
A decorator intercepts invocations from a specific Java interface, and is aware of all the semantics attached to that interface. Decorators are useful for modeling some kinds of business concerns, but do not have the generality of interceptors. A decorator is a bean, or even an abstract class, that implements the type it decorates, and is annotated with @Decorator
. To invoke a decorator in a CDI application, it must be specified in the beans.xml
file.
Example: Invoke a Decorator Through beans.xml
This declaration serves two main purposes:
- It enables us to specify an ordering for decorators in our system, ensuring deterministic behavior
- It lets us enable or disable decorator classes at deployment time.
A decorator must have exactly one @Delegate
injection point to obtain a reference to the decorated object.
Example: Decorator Class
From CDI 1.1 onwards, the decorator can be enabled for the whole application using @Priority
annotation.
Decorators enabled using @Priority
are called before decorators enabled using beans.xml
. The lower priority values are called first.
Having a decorator enabled by @Priority
and at the same time invoked by beans.xml
, leads to a non-portable behavior. This combination of enablement should therefore be avoided in order to maintain consistent behavior across different CDI implementations.
7.13. Portable Extensions Copier lienLien copié sur presse-papiers!
CDI is intended to be a foundation for frameworks, extensions, and for integration with other technologies. Therefore, CDI exposes a set of SPIs for the use of developers of portable extensions to CDI.
Extensions can provide the following types of functionality:
- Integration with Business Process Management engines
- Integration with third-party frameworks, such as Spring, Seam, GWT, or Wicket
- New technology based upon the CDI programming model
According to the JSR-346 specification, a portable extension can integrate with the container in the following ways:
- Providing its own beans, interceptors, and decorators to the container
- Injecting dependencies into its own objects using the dependency injection service
- Providing a context implementation for a custom scope
- Augmenting or overriding the annotation-based metadata with metadata from another source
7.14. Bean Proxies Copier lienLien copié sur presse-papiers!
Clients of an injected bean do not usually hold a direct reference to a bean instance. Unless the bean is a dependent object, scope @Dependent
, the container must redirect all injected references to the bean using a proxy object.
A bean proxy, which can be referred to as client proxy, is responsible for ensuring the bean instance that receives a method invocation is the instance associated with the current context. The client proxy also allows beans bound to contexts, such as the session context, to be serialized to disk without recursively serializing other injected beans.
Due to Java limitations, some Java types cannot be proxied by the container. If an injection point declared with one of these types resolves to a bean with a scope other than @Dependent
, the container aborts the deployment.
Certain Java types cannot be proxied by the container. These include:
- Classes that do not have a non-private constructor with no parameters
-
Classes that are declared
final
or have afinal
method - Arrays and primitive types
7.15. Use a Proxy in an Injection Copier lienLien copié sur presse-papiers!
A proxy is used for injection when the lifecycles of the beans are different from each other. The proxy is a subclass of the bean that is created at run-time, and overrides all the non-private methods of the bean class. The proxy forwards the invocation onto the actual bean instance.
In this example, the PaymentProcessor
instance is not injected directly into Shop
. Instead, a proxy is injected, and when the processPayment()
method is called, the proxy looks up the current PaymentProcessor
bean instance and calls the processPayment()
method on it.
Example: Proxy Injection