原文
第一种配置方法:使用@AspectJ标签
- 在配置文件中添加<aop:aspectj-autoproxy/>注解
- 创建一个Java文件,使用@Aspect注解修饰该类
- 创建一个方法,使用@Before、@After、@Around等进行修饰,在注解中写上切入点的表达式
说明:上述Java文件创建好后,需要将其在Spring的容器中进行声明,可以在配置文件中定义<bean/>节点,也可以使用@Component组件进行修饰
示例:
- import org.aspectj.lang.ProceedingJoinPoint;
- import org.aspectj.lang.annotation.After;
- importorg.aspectj.lang.annotation.AfterThrowing;
- import org.aspectj.lang.annotation.Around;
- import org.aspectj.lang.annotation.Aspect;
- import org.aspectj.lang.annotation.Before;
- importorg.springframework.stereotype.Component;
- @Component
- @Aspect
- public class AopLog {
- //方法执行前调用
- @Before("execution (*com.zywang.services.impl.*.*(..))")
- public void before() {
- System.out.println("before");
- }
- //方法执行后调用
- @After("execution (*com.zywang.services.impl.*.*(..))")
- public void after() {
- System.out.println("after");
- }
- //方法执行的前后调用
- @Around("execution (*com.zywang.services.impl.*.*(..))")
- public Object around(ProceedingJoinPoint point)throws Throwable{
- System.out.println("begin around");
- Object object = point.proceed();
- System.out.println("end around");
- return object;
- }
- //方法运行出现异常时调用
- @AfterThrowing(pointcut = "execution (*com.zywang.services.impl.*.*(..))",throwing ="ex")
- public void afterThrowing(Exceptionex){
- System.out.println("afterThrowing");
- System.out.println(ex);
- }
- }
import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.AfterThrowing; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.springframework.stereotype.Component; @Component @Aspect public class AopLog { //方法执行前调用 @Before("execution (* com.zywang.services.impl.*.*(..))") public void before() { System.out.println("before"); } //方法执行后调用 @After("execution (* com.zywang.services.impl.*.*(..))") public void after() { System.out.println("after"); } //方法执行的前后调用 @Around("execution (* com.zywang.services.impl.*.*(..))") public Object around(ProceedingJoinPoint point) throws Throwable{ System.out.println("begin around"); Object object = point.proceed(); System.out.println("end around"); return object; } //方法运行出现异常时调用 @AfterThrowing(pointcut = "execution (* com.zywang.services.impl.*.*(..))",throwing = "ex") public void afterThrowing(Exception ex){ System.out.println("afterThrowing"); System.out.println(ex); } }
上面这段代码中多次使用了重复的切入点,这种情况下,可以使用@Pointcut标注,来修改一个切入点方法(这个方法不需要参数和方法体),然后就可以在@Before等标注中引用该方法作为切入点,示例如下:
- import org.aspectj.lang.ProceedingJoinPoint;
- import org.aspectj.lang.annotation.Around;
- import org.aspectj.lang.annotation.Aspect;
- import org.aspectj.lang.annotation.Before;
- import org.aspectj.lang.annotation.Pointcut;
- importorg.springframework.stereotype.Component;
- @Component
- @Aspect
- public class AopLog {
- @Pointcut("execution (*com.iflysse.school.services.impl.*.*(..))")
- public void pointcut(){}
- //方法执行前调用
- @Before("pointcut()")
- public void before() {
- System.out.println("before");
- }
- //方法执行的前后调用
- @Around("pointcut()")
- public Object around(ProceedingJoinPoint point)throws Throwable{
- System.out.println("begin around");
- Object object = point.proceed();
- System.out.println("end around");
- return object;
- }
- }
import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; @Component @Aspect public class AopLog { @Pointcut("execution (* com.iflysse.school.services.impl.*.*(..))") public void pointcut(){} //方法执行前调用 @Before("pointcut()") public void before() { System.out.println("before"); } //方法执行的前后调用 @Around("pointcut()") public Object around(ProceedingJoinPoint point) throws Throwable{ System.out.println("begin around"); Object object = point.proceed(); System.out.println("end around"); return object; } }
第二种配置方法:基于配置文件的配置
- 创建一个Java文件,并指定一个用于执行拦截的方法,该方法可以有0个或多个参数
- 在Spring配置文件中注册该Java类为一个Bean
- 使用<aop:config/>、<aop:aspect/>等标签进行配置
示例:
Java文件
- import org.aspectj.lang.ProceedingJoinPoint;
- public class AopLog {
- //方法执行的前后调用
- public Object runOnAround(ProceedingJoinPointpoint) throws Throwable{
- System.out.println("begin around");
- Object object = point.proceed();
- System.out.println("end around");
- return object;
- }
- }
import org.aspectj.lang.ProceedingJoinPoint; public class AopLog { //方法执行的前后调用 public Object runOnAround(ProceedingJoinPoint point) throws Throwable{ System.out.println("begin around"); Object object = point.proceed(); System.out.println("end around"); return object; } }
Spring配置文件
- <beanid="aopLog"class="com.iflysse.school.aop.AopLog"></bean>
- <aop:config>
- <aop:aspectref="aopLog">
- <aop:aroundmethod="runOnAround"pointcut="execution (*com.zywang.services.impl.*.*(..))"/>
- </aop:aspect>
- </aop:config>
<!-- 声明式事务控制 -->
<bean id="txManager"
class="org.springframework.orm.hibernate3.HibernateTransactionMana
ger">
<property name="sessionFactory"ref="mySessionFactory">
</property>
</bean>
<tx:advice id="txAdvice"transaction-manager="txManager">
<tx:attributes>
<tx:method name="save*" propagation="REQUIRED"/>
<tx:method name="update*"propagation="REQUIRED" />
<tx:method name="delete*"propagation="REQUIRED" />
<tx:method name="find*" read-only="true"
propagation="NOT_SUPPORTED"/>
<tx:method name="*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcutexpression="within(com.michael.service.UserService..*)"id="servicePointcut" />
<aop:advisor advice-ref="txAdvice"pointcut-ref="servicePointcut" />
</aop:config>
<bean id="aopLog" class="com.iflysse.school.aop.AopLog"></bean> <aop:config> <aop:aspect ref="aopLog"> <aop:around method="runOnAround" pointcut="execution (* com.zywang.services.impl.*.*(..))"/> </aop:aspect> </aop:config>
注意:上面这个示例使用的是around方式的拦截,该方法要求Java类中的方法有一个ProceedingJoinPoint类型的参数
使用第二种方式的AOP配置,在Eclipse(有SpringIDE插件)中被拦截到的方法中有标识显示
以上配置基于Spring 3.0.5 进行设置,参考其《ReferenceDocumentation》