Spring AOP 后篇(三): AOP切面编程

Spring AOP 后篇: AOP切面编程

该文章参考多篇文章的基础上进行了简化并做少许修改,方便理解。原文章地址如下:

  1. Spring3:AOP
  2. Spring的AOP面向切面编程

一、理解切入点表达式(execution())(重点)

  • 切入点指示符

    • execution:用于匹配方法执行连接点。这是使用Spring AOP时使用的主要切入点指示符。
    • within:限制匹配某些类型中的连接点(使用Spring AOP时在匹配类型中声明的方法的执行)。
    • this:限制与连接点的匹配(使用Spring AOP时执行方法),其中bean引用(Spring AOP代理)是给定类型的实例。
    • target:限制匹配连接点(使用Spring AOP时执行方法),其中目标对象(被代理的应用程序对象)是给定类型的实例。
    • args:限制与连接点的匹配(使用Spring AOP时执行方法),其中参数是给定类型的实例。
    • @target:限制与连接点的匹配(使用Spring AOP时执行方法),其中执行对象的类具有给定类型的注释。
    • @args:限制与连接点的匹配(使用Spring AOP时执行方法),其中传递的实际参数的运行时类型具有给定类型的注释。
    • @within:限制匹配到具有给定注释的类型中的连接点(使用Spring AOP时执行在具有给定注释的类型中声明的方法)。
    • @annotation:限制连接点的匹配,其中连接点的主题(在Spring AOP中执行的方法)具有给定的注释。

    within、this、args等详细解释。。。(todo)

  • execution表达式 详解参考Spring AOP 官方文档

    <!-- * 号表示通配符; 
    如下 execution表达式表示 匹配service包下的所有类的所有带参及不带参数的方法 -->
    <aop:pointcut id="businessServcie" 
        expression="execution(* com.xyz.myapp.service.*.*(..))" />
    
    1. execution() : 表达式主体
    2. 第一个’*'号位置:表示 返回类型;*作用是匹配任何返回类型;
    3. com.xyz.myapp.service: 表示包名,即service包
    4. 第二个’*'号位置:表示 类名;*作用是匹配该包下的所有类
    5. 第三个‘*’号位置:表示 方法名; *作用是匹配所有方法
    6. (…): 表示 括号里面代表参数;()表示匹配一个不带参数的方法;(…)表示匹配任何数量(零个或多个)参数;(*)表示匹配一个采用任何类型参数的方法

切入点其他表达式

  • 常用切入点表达式(execution())Spring AOP官方文档

    1. 执行任何公共方法:
    execution(public * *(..))
    
    1. 执行名称以以下开头的任何方法set
    execution(* set*(..))
    
    1. 执行AccountService接口定义的任何方法:
    execution(* com.xyz.service.AccountService.*(..))
    
    1. 执行service包中定义的任何方法:
    execution(* com.xyz.service.*.*(..))
    
    1. 执行服务包或其子包中定义的任何方法:
    execution(* com.xyz.service..*.*(..))
    
    1. 服务包中的任何连接点(仅在Spring AOP中执行方法):
    within(com.xyz.service.*)
    
    1. 服务包或其子包中的任何连接点(仅在Spring AOP中执行方法):
    within(com.xyz.service..*)
    
    1. 代理实现AccountService接口的任何连接点(仅在Spring AOP中执行方法) :
    this(com.xyz.service.AccountService)
    

二、XML手动配置方式

  1. aopBean.xml
<?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"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop-4.2.xsd">
        
    	<!-- dao 实例 -->
        <bean id="userDao" class="com.aop.UserDao" />
        <bean id="orderDao" class="com.aop.OrderDao" />
    	<!-- 切面类:在调用dao类时,先后调用 时间打印及日志打印类 -->
        <bean id="timePrint" class="com.aop.TimePrint" />
        <bean id="logPrint" class="com.aop.LogPrint" />
        
        <aop:config>
            <!-- 定义切面类
			ref: 指定切面类; 
			order(int):指定切面的优先级来控制通知的执行顺序; order1 》order2
 			-->
            <aop:aspect id="time" ref="timePrint" order="1">
                <!-- 切入点
 				expression: 定义切入点表达式;即定义要切入的类或方法
				-->
                <aop:pointcut id="addTime" expression="execution(* com.aop.userDao.*(..))" />
                <!-- 前置通知: 在目标方法调用前执行 
				method:切面类执行的方法;即目标类调用前执行的方法
				-->
                <aop:before method="printTime" pointcut-ref="addTime" />
                <!-- 后置通知: 在目标方法调用后执行 -->
                <aop:after method="printTime" pointcut-ref="addTime" />
            </aop:aspect>
            
            <aop:aspect id="log" ref="logPrint" order="2">
                <aop:pointcut id="printLog" expression="execution(* com.aop.*.*(..))" />
                <aop:before method="LogBefore" pointcut-ref="printLog" />
                <aop:after method="LogAfter" pointcut-ref="printLog" />
            </aop:aspect>
        </aop:config>
</beans>
  1. aopBean.xml

    <?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:p="http://www.springframework.org/schema/p"
        xmlns:context="http://www.springframework.org/schema/context"
        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/context
            http://www.springframework.org/schema/context/spring-context.xsd
            http://www.springframework.org/schema/aop
            http://www.springframework.org/schema/aop/spring-aop.xsd">
        
        <!-- dao 实例   -->
        <bean id="userDao" class="com.aop.dao.UserDao"></bean>
        <bean id="orderDao" class="com.aop.dao.OrderDao"></bean>
        
        <!-- 切面类 -->
        <bean id="aop" class="com.aop.dao.Aop"></bean>
        
        <!-- Aop配置 -->
        <aop:config>
            <!-- 定义一个切入点表达式: 拦截哪些方法 -->
            <aop:pointcut  id="pt" expression="execution(* com.aop.dao.*.*(..))"/>
            <!-- 切面 -->
            <aop:aspect ref="aop">
                <!-- 环绕通知 -->
                <aop:around method="around" pointcut-ref="pt"/>
                <!-- 前置通知: 在目标方法调用前执行 -->
                <aop:before method="begin" pointcut-ref="pt"/>
                <!-- 后置通知: -->
                <aop:after method="after" pointcut-ref="pt"/>
                <!-- 返回后通知 -->
                <aop:after-returning method="afterReturning" pointcut-ref="pt"/>
                <!-- 异常通知 -->
                <aop:after-throwing method="afterThrowing" pointcut-ref="pt"/>
                
            </aop:aspect>
        </aop:config>
    </beans>
    

三、@Aspectj 注解方式

  1. bean.xml 配置aop注解扫描

    <?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:p="http://www.springframework.org/schema/p"
        xmlns:context="http://www.springframework.org/schema/context"
        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/context
            http://www.springframework.org/schema/context/spring-context.xsd
            http://www.springframework.org/schema/aop
          http://www.springframework.org/schema/aop/spring-aop.xsd">
        
        <!-- 使用注解时要开启注解扫描 要扫描的包 -->
        <context:component-scan base-package="cn.itcast.e_aop_anno"></context:component-scan>
        
        <!-- 开启aop注解方式 -->
        <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
    </beans>
    
    1. 使用切面类使用@Aspect 注解

      @Component  //加入IOC容器
      @Aspect  // 指定当前类为切面类
      public class Aop {
      
          // 指定切入点表达式: 拦截哪些方法; 即为哪些类生成代理对象
        //解释@Pointcut("execution(* cn.itcast.e_aop_anno.*.*(..))")
        //@Pointcut("execution(*    切入点表达式固定写法, cn.itcast.e_aop_anno表示包.类名(可以用*表示包下所有的类).方法名(可以用*表示类下所有的方法)(..)表示参数可以用..
          @Pointcut("execution(* cn.itcast.e_aop_anno.*.*(..))")
          public void pointCut_(){
          }
          
        //@Before("execution(* cn.itcast.e_aop_anno.*.*(..))")每个方法需要写相同的引用,所以将相同的部分抽取到一个空的方法中pointCut_(),
          // 前置通知 : 在执行目标方法之前执行
          @Before("pointCut_()")
          public void begin(){
              System.out.println("开始事务/异常");
          }
          
          // 后置/最终通知:在执行目标方法之后执行  【无论是否出现异常最终都会执行】
          @After("pointCut_()")
          public void after(){
              System.out.println("提交事务/关闭");
          }
          
          // 返回后通知: 在调用目标方法结束后执行 【出现异常不执行】
          @AfterReturning("pointCut_()")
          public void afterReturning() {
              System.out.println("afterReturning()");
          }
          
          // 异常通知: 当目标方法执行异常时候执行此关注点代码
          @AfterThrowing("pointCut_()")
          public void afterThrowing(){
              System.out.println("afterThrowing()");
          }
          
          // 环绕通知:环绕目标方式执行
          @Around("pointCut_()")
          public void around(ProceedingJoinPoint pjp) throws Throwable{
              System.out.println("环绕前....");
              pjp.proceed();  // 执行目标方法
              System.out.println("环绕后....");
          }
          
      }
      

猜你喜欢

转载自blog.csdn.net/StarryaSky/article/details/84038529