Spring AOP - Implémentations

Spring soutient le @AspectJ annotation style approche et le schema-based approche pour implémenter les aspects personnalisés.

Basé sur un schéma XML

Les aspects sont implémentés en utilisant des classes régulières avec une configuration basée sur XML.

Pour utiliser les balises d'espace de noms AOP décrites dans cette section, vous devez importer le schéma Spring AOP, décrit comme suit:

<?xml version = "1.0" encoding = "UTF-8"?>
<beans xmlns = "http://www.springframework.org/schema/beans"
   
   xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance" 
   xmlns:aop = "http://www.springframework.org/schema/aop"
   xsi:schemaLocation = "http://www.springframework.org/schema/beans
   http://www.springframework.org/schema/beans/spring-beans-3.0.xsd 
   http://www.springframework.org/schema/aop 
   http://www.springframework.org/schema/aop/spring-aop-3.0.xsd ">

   <!-- bean definition & AOP specific configuration -->

</beans>

Déclarer un aspect

Un aspect est déclaré en utilisant le <aop:aspect> élément, et le bean de support est référencé à l'aide de ref attribut comme suit.

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">
   ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

Ici, "aBean" sera configuré et la dépendance injectée comme n'importe quel autre bean Spring comme vous l'avez vu dans les chapitres précédents.

Déclarer un PointCut

UNE PointCutaide à déterminer les points de jonction (c'est-à-dire les méthodes) d'intérêt à exécuter avec différents conseils. Lorsque vous travaillez avec une configuration basée sur un schéma XML, PointCut sera défini comme suit -

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">

   <aop:PointCut id = "businessService"
      expression = "execution(* com.xyz.myapp.service.*.*(..))"/>
      ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

L'exemple suivant définit un PointCut nommé 'businessService' qui correspondra à l'exécution de la méthode getName () disponible dans la classe Student sous le package com.tutorialspoint.

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">

   <aop:PointCut id = "businessService"
      expression = "execution(* com.tutorialspoint.Student.getName(..))"/>
   ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

Déclarer des avis

Vous pouvez déclarer n'importe lequel des cinq conseils dans un <aop: aspect> en utilisant l'élément <aop: {ADVICE NAME}> comme suit.

<aop:config>
   <aop:aspect id = "myAspect" ref = "aBean">
      <aop:PointCut id = "businessService"
         expression = "execution(* com.xyz.myapp.service.*.*(..))"/>

      <!-- a before advice definition -->
      <aop:before PointCut-ref = "businessService" 
         method = "doRequiredTask"/>

      <!-- an after advice definition -->
      <aop:after PointCut-ref = "businessService" 
         method = "doRequiredTask"/>

      <!-- an after-returning advice definition -->
      <!--The doRequiredTask method must have parameter named retVal -->
      <aop:after-returning PointCut-ref = "businessService"
         returning = "retVal"
         method = "doRequiredTask"/>

      <!-- an after-throwing advice definition -->
      <!--The doRequiredTask method must have parameter named ex -->
      <aop:after-throwing PointCut-ref = "businessService"
        throwing = "ex"
         method = "doRequiredTask"/>

      <!-- an around advice definition -->
      <aop:around PointCut-ref = "businessService" 
         method = "doRequiredTask"/>
   ...
   </aop:aspect>
</aop:config>

<bean id = "aBean" class = "...">
   ...
</bean>

Vous pouvez utiliser le même doRequiredTaskou différentes méthodes pour différents conseils. Ces méthodes seront définies dans le cadre du module aspect.

@AspectJ basé

@AspectJ fait référence à un style de déclaration d'aspects en tant que classes Java régulières annotées avec des annotations Java 5. La prise en charge de @AspectJ est activée en incluant l'élément suivant dans votre fichier de configuration basé sur un schéma XML.

<aop:aspectj-autoproxy/>

Déclarer un aspect

Les classes Aspects sont comme n'importe quel autre bean normal et peuvent avoir des méthodes et des champs comme n'importe quelle autre classe, sauf qu'ils seront annotés avec @Aspect comme suit.

package org.xyz;

import org.aspectj.lang.annotation.Aspect;

@Aspect
public class AspectModule {

}

Ils seront configurés en XML comme tout autre bean comme suit.

<bean id = "myAspect" class = "org.xyz.AspectModule">
   <!-- configure properties of aspect here as normal -->
</bean>

Déclarer un PointCut

UNE PointCutaide à déterminer les points de jonction (c'est-à-dire les méthodes) d'intérêt à exécuter avec différents conseils. Lorsque vous travaillez avec une configuration basée sur @AspectJ, la déclaration PointCut comporte deux parties -

  • Une expression PointCut qui détermine exactement les exécutions de méthode qui nous intéressent.

  • Une signature PointCut comprenant un nom et un nombre quelconque de paramètres. Le corps réel de la méthode n'est pas pertinent et devrait en fait être vide.

L'exemple suivant définit un PointCut nommé 'businessService' qui correspondra à l'exécution de chaque méthode disponible dans les classes sous le package com.xyz.myapp.service.

import org.aspectj.lang.annotation.PointCut;

@PointCut("execution(* com.xyz.myapp.service.*.*(..))") // expression 
private void businessService() {}  // signature

L'exemple suivant définit un PointCut nommé 'getname' qui correspondra à l'exécution de la méthode getName () disponible dans la classe Student sous le package com.tutorialspoint.

import org.aspectj.lang.annotation.PointCut;

@PointCut("execution(* com.tutorialspoint.Student.getName(..))") 
private void getname() {}

Déclarer des avis

Vous pouvez déclarer l'un des cinq avis en utilisant les annotations @ {ADVICE-NAME} comme indiqué ci-dessous. Cela suppose que vous avez déjà défini une méthode de signature PointCut businessService ().

@Before("businessService()")
public void doBeforeTask(){
   ...
}

@After("businessService()")
public void doAfterTask(){
   ...
}

@AfterReturning(PointCut = "businessService()", returning = "retVal")
public void doAfterReturnningTask(Object retVal){
   // you can intercept retVal here.
   ...
}

@AfterThrowing(PointCut = "businessService()", throwing = "ex")
public void doAfterThrowingTask(Exception ex){
   // you can intercept thrown exception here.
   ...
}

@Around("businessService()")
public void doAroundTask(){
   ...
}

Vous pouvez définir PointCut en ligne pour n'importe lequel des conseils. Voici un exemple de définition de PointCut en ligne pour les conseils avant.

@Before("execution(* com.xyz.myapp.service.*.*(..))")
public doBeforeTask(){
   ...
}