除了使用AspectJ注解声明切面,Spring也支持在bean配置文件中声明切面。这种声明是通过aop名称空间中的XML元素完成的。正常情况下,基于注解的声明要优先于基于XML的声明。通过AspectJ注解,切面可以与AspectJ兼容,而基于XML的配置则是Spring专有的。由于AspectJ得到越来越多的 AOP框架支持,所以以注解风格编写的切面将会有更多重用的机会。

一、配置细节

在bean配置文件中,所有的Spring AOP配置都必须定义在<aop:config>元素内部对于每个切面而言,都要创建一个<aop:aspect>元素来为具体的切面实现引用后端bean实例。

切面bean必须有一个标识符,供<aop:aspect>元素引用。

<bean id="calculatorLoggingAspect" class="com.jdy.spring2020.aop.CalculatorLoggingAspect"/>
<aop:config>
    <aop:aspect id="calculatorLoggingAspect" order="0" ref="calculatorLoggingAspect">
    </aop:aspect>
</aop:config>

二、xml配置

  1. 切入点使用<aop:pointcut>元素声明。
  2. 切入点必须定义在<aop:aspect>元素下,或者直接定义在<aop:config>元素下。
  • 定义在<aop:aspect>元素下:只对当前切面有效
  • 定义在<aop:config>元素下:对所有切面都有效
  1. 基于XML的AOP配置不允许在切入点表达式中用名称引用其他切入点。
  2. 在aop名称空间中,每种通知类型都对应一个特定的XML元素。
  3. 通知元素需要使用<pointcut-ref>来引用切入点,或用<pointcut>直接嵌入切入点表达式。
  4. method属性指定切面类中通知方法的名称
<?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.xsd
      http://www.springframework.org/schema/aop
      https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--定义目标类-->
    <bean id="arithmeticCalculator" class="com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl"/>
    <!--定义切面类-->
    <bean name="arithmeticCalculatorLog" class="com.jdy.spring2020.aop.ArithmeticCalculatorLog"/>

    <!--aop配置-->
    <aop:config>
        <!--定义切点,其中匹配的是com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl类下面所有参数返回值任意的方法-->
            <aop:pointcut id="pointcut"
                      expression="execution(* com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.*(..))"/>
        <aop:aspect  ref="arithmeticCalculatorLog">
            <!--前置通知,method必须与切面类中的方法名相同(下同),并指定切点-->
            <aop:before method="before" pointcut-ref="pointcut"></aop:before>
            <!--后置通知-->
            <aop:after method="after" pointcut-ref="pointcut"></aop:after>
            <!--返回通知,返回值名与方法返回值名相同-->
            <aop:after-returning method="afterRunning" pointcut-ref="pointcut" returning="result"/>
            <!--异常通知,异常名与方法返回的异常名相同-->
            <aop:after-throwing method="afterThrowing" pointcut-ref="pointcut" throwing="ex"/>
        </aop:aspect>
    </aop:config>
</beans>

三、切面类

package com.jdy.spring2020.aop;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;

public class ArithmeticCalculatorLog {

    /**
     * 前置通知:
     * @Before("execution( com.jdy.spring2020.scan.service.impl.ArithmeticCalculatorImpl.add(int,int))")
     * 在ArithmeticCalculatorImpl.add方法前之前前置通知
     *
     */
    public void before(){
        System.out.println("AO....前置通知");
    }

    /**
     *
     * @param joinPoint:连接点对象
     */
    public void after(JoinPoint joinPoint){
        //方法的名字
        String name = joinPoint.getSignature().getName();
        System.out.println("后置通知方法名字:" + name);
    }

    /**
     * 返回通知:目标方法执行结束后,得到方法的返回值
     * 获取方法的返回值:通过returnning来指定一个名字,必须要与方法的一个参数名成一致。
     * @param joinPoint
     * @param result
     */
    public void afterRunning(JoinPoint joinPoint,Object result){
        //方法的名字
        String name = joinPoint.getSignature().getName();
        System.out.println("返回通知方法名字:" + name);
        System.out.println("返回通知方法返回值:" + result);
    }


    /**
     * 异常通知:目标方法执行结束后,目标方法抛出异常
     * 获取方法的异常:通过Throwing来指定一个名字,必须要与方法的一个参数名一致
     * 可以通过形参中异常的类型来设置抛出指定异常才会执行异常通知
     * @param joinPoint
     */
    public void afterThrowing(JoinPoint joinPoint,Exception ex){
        //方法的名字
        String name = joinPoint.getSignature().getName();
        System.out.println("异常通知方法名字:" + name);
        System.out.println("异常通知方法返回异常:" + ex);
    }

    //可以通过形参中异常的类型来设置抛出指定异常才会执行异常通知
    public void afterThrowing1(JoinPoint joinPoint, NullPointerException ex){
        //方法的名字
        String name = joinPoint.getSignature().getName();
        System.out.println("异常通知方法名字" + name);
        System.out.println("异常通知方法返回异常" + ex);
    }


    /**
     * 环绕通知
     * @param joinPoint
     */
    public Object around(ProceedingJoinPoint joinPoint) {
        //方法的名字
        Object obj = null;
        //前置
        try {
            //执行目标方法,相当于动态代理的invoke方法
            System.out.println("环绕通知---->方法名字" + joinPoint.getSignature().getName());
            obj = joinPoint.proceed();
            System.out.println("环绕通知---->方法proceed" + obj);
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }finally {
            //后置
        }
        return obj;
    }
}