zoukankan      html  css  js  c++  java
  • spring AOP 编程--AspectJ注解方式 (4)

    1. AOP 简介

    AOP(Aspect-Oriented Programming, 面向切面编程): 是一种新的方法论, 是对传统 OOP(Object-Oriented Programming, 面向对象编程) 的补充.

    AOP 的主要编程对象是切面(aspect), 而切面模块化横切关注点.

    在应用 AOP 编程时, 仍然需要定义公共功能, 但可以明确的定义这个功能在哪里, 以什么方式应用, 并且不必修改受影响的类. 这样一来横切关注点就被模块化到特殊的对象(切面)里.

    AOP 的好处:

    ---每个事物逻辑位于一个位置, 代码不分散, 便于维护和升级

    ---业务模块更简洁, 只包含核心业务代码.

    2. AOP的术语和图形表达

    切面(Aspect): 横切关注点(跨越应用程序多个模块的功能)被模块化的特殊对象

    通知(Advice): 切面必须要完成的工作

    目标(Target): 被通知的对象

    代理(Proxy): 向目标对象应用通知之后创建的对象

    连接点(Joinpoint):程序执行的某个特定位置:如类某个方法调用前、调用后、方法抛出异常后等。连接点由两个信息确定:方法表示的程序执行点;相对点表示的方位。例如 ArithmethicCalculator#add() 方法执行前的连接点,执行点为 ArithmethicCalculator#add(); 方位为该方法执行前的位置

    切点(pointcut):每个类都拥有多个连接点:例如 ArithmethicCalculator 的所有方法实际上都是连接点,即连接点是程序类中客观存在的事务。AOP 通过切点定位到特定的连接点。类比:连接点相当于数据库中的记录,切点相当于查询条件。切点和连接点不是一对一的关系,一个切点匹配多个连接点,切点通过 org.springframework.aop.Pointcut 接口进行描述,它使用类和方法作为连接点的查询条件。

    通知:

      前置通知(before advice):在切入点之前执行。

      后置通知(after returning advice):在切入点执行完成后,执行通知。无论是否抛出异常,都会执行。在后置通知中不能访问方法返回的结果

      环绕通知(around advice):包围切入点,调用方法前后完成自定义行为。

      异常通知(after throwing advice):在切入点抛出异常后,执行通知。

    方式一、aspectj注解

    1、需要引入一个

    2、ArithmeticCalculator.java

    复制代码
    1 package com.proc;
    2 
    3 public interface ArithmeticCalculator {
    4     int add(int i, int j);
    5     int sub(int i, int j);
    6     
    7     int mul(int i, int j);
    8     int div(int i, int j);
    9 }
    复制代码

     3、ArithmeticCalculatorImpl.java

    复制代码
     1 package com.proc;
     2 import org.springframework.stereotype.Component;
     3 
     4 
     5 @Component("arithmeticCalculator")
     6 public class ArithmeticCalculatorImpl implements ArithmeticCalculator{
     7     public int add(int i, int j) {
     8         int result = i + j;
     9         return result;
    10     }
    11 
    12     public int sub(int i, int j) {
    13         int result = i - j;
    14         return result;
    15     }
    16 
    17     public int mul(int i, int j) {
    18         int result = i * j;
    19         return result;
    20     }
    21 
    22     public int div(int i, int j) {
    23         int result = i / j;
    24         return result;
    25     }
    26 }
    复制代码

     4、LoggingAspect.java

    复制代码
     1 package com.proc;
     2 
     3 import org.aspectj.lang.JoinPoint;
     4 import org.aspectj.lang.annotation.Aspect;
     5 import org.aspectj.lang.annotation.Before;
     6 import org.springframework.stereotype.Component;
     7 
     8 @Aspect
     9 @Component
    10 public class LoggingAspect {
    11 
    12     @Before("execution(* *.*(int,int))")
    13     public void beforeMethod(JoinPoint point){
    14         System.out.println("正在执行方法: "+point.getSignature().getName());
    15     }
    16 }
    复制代码

      这是一个日志切面类,该类首先是一个IoC中的bean,通过@Component将bean交个容器管理,然后@Aspect声明这是一个切面。

      @Before是通知时机,表示在调用方法之前通知。 里面字符串格式为execution (访问修饰符 返回类型 包名.类名.方法名(参数类型...))。

      beforeMethod是定一个的一个切面执行的方法,方法名可以任意指定。可以不带参数,也可以指定一个JoinPoint连接点的参数。

    配置

    1 <context:component-scan base-package="com.proc" />
    2 <aop:aspectj-autoproxy></aop:aspectj-autoproxy>

       aop:aspectj-autoproxy:使得@Aspect注解生效

    测试代码:

    复制代码
    1 ApplicationContext ctx=new ClassPathXmlApplicationContext("applicationContext.xml");
    2 ArithmeticCalculator calc=(ArithmeticCalculator) ctx.getBean("arithmeticCalculator");
    3 System.out.println(calc.add(3, 5));
    4 System.out.println(calc.sub(3, 5)); 
    5 System.out.println(calc.mul(6, 2)); 
    6 System.out.println(calc.div(6, 2)); 
    复制代码

    输出结果

    复制代码
    正在执行方法add
    8
    正在执行方法sub
    -2
    正在执行方法mul
    12
    正在执行方法div
    3
    复制代码

    上面方法是用的前置通知。下面我们讲解一下后置通知、返回通知、异常通知和环绕通知

    ☆后置通知

      后置通知无论方法是否抛出异常,都会执行。在后置通知中不能访问返回结果,因为可能会出现异常,就没有返回值

    1 @After("execution(* *.*(int,int))")
    2 public void afterMethod(JoinPoint point){
    3     System.out.println("方法执行结束: "+point.getSignature().getName());
    4 }

    ☆返回通知

      当方法有异常时,不会执行返回通知,只有在方法正常执行结束才能够执行返回通知。在结果通知中用参数returning来指定接收返回结果对象的名称,直接传递给下面方法中对应的参数

    1 @AfterReturning(value="execution(* *.*(int,int))",returning="retVal")
    2 public void afterReturningMethod(JoinPoint point,Object retVal){
    3     System.out.println("方法: "+point.getSignature().getName()+"执行结果为:"+retVal);
    4 }

    ☆异常通知

      当方法有异常时,才会执行返回异常通知

    1 @AfterThrowing(value="execution(* *.*(int,int))",throwing="ex")
    2 public void afterThrowingMethod(JoinPoint point,Exception ex){
    3     System.out.println("执行方法: "+point.getSignature().getName()+"出现了异常:"+ex.getMessage());
    4 }

    【注意】:在指定异常类型时,只能捕获到该类型或子类型的异常。比如指定NullPointerException异常,就只能捕获空指针异常,所以,我们可以指定为异常的父类型。

    ☆环绕通知

      环绕通知是一个比较强大,但相对复杂一点的通知,我们通常不使用

    复制代码
     1 @Around("execution(* *.*(int,int))")
     2 public Object aroundMethod(ProceedingJoinPoint point){
     3     
     4     System.out.println("环绕通知: "+point.getSignature().getName());
     5     Object result=null;
     6     //这里相当于前置通知
     7     try {
     8         //执行方法
     9         result= point.proceed();
    10         //这里相当于结果通知
    11     } catch (Throwable e) {
    12         //这里相当于异常通知
    13         e.printStackTrace();
    14         
    15     }
    16     //这里相当于后置通知
    17     System.out.println("环绕通知: "+point.getSignature().getName());
    18     return result;
    19 }
    复制代码

    【注意】:环绕通知要求有一个返回值point.proceed()执行方法,并提供一个返回值

  • 相关阅读:
    怎么说???
    再给自己一点鼓励
    还是不要看新闻的好
    系统架构师
    朋友
    未来的路
    I now have a dream
    纪念死去的手机
    全局规划
    终于要上战场了!
  • 原文地址:https://www.cnblogs.com/weiqingfeng/p/9497975.html
Copyright © 2011-2022 走看看