欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

深入理解spring的AOP机制原理

程序员文章站 2024-03-31 12:46:16
前言 在软件开发中,散布于应用中多处的功能被称为横切关注点,通常来讲,这些横切关注点从概念上是与应用的业务逻辑相分离的。把这些横切关注点和业务逻辑分离出来正是aop要...

前言

在软件开发中,散布于应用中多处的功能被称为横切关注点,通常来讲,这些横切关注点从概念上是与应用的业务逻辑相分离的。把这些横切关注点和业务逻辑分离出来正是aop要解决的问题。aop能够帮我们模块化横切关注点,换言之,横切关注点可以被描述为影响应用多出的功能。这些横切点被模块化特殊的类,这些类被称为切面。

术语定义

通知:切面有必须要完成的工作,在aop中,切面的工作被称为通知。通知定义了切面是什么以及何时使用,除了描述切面要完成的工作,通知还解决了何时执行这个工作的问题,它应该在某个方法之前?之后?之前和之后都调用?还是只在方法抛出异常时调用?

连接点:连接点是应用程序执行过程中,能够插入切面的一个点。

切点:是在连接点的基础上定义切点,比方说一个类由十几个方法,每个方法的调用前和调用后都可以插入通知,但是你只想选择几个方法插入通知,因此你定义一个切点来选择你想插入的通知的方法。

切面:切面就是通知和切点的结合。

织入:织入是把切面应用到目标对象并创建新的代理对象的过程,切面在指定的连接点被织入到目标对象中。在目标对象的生命周期里有多个点可以进行织入:编译期、类加载期、运行期。其中编译器织入需要特殊的编译器,类加载器织入需要特殊的类加载器,spring的aop 是在运行期织入通知的。

spring的aop支持

spring提供了aop的四种支持,分别是:基于代理的经典spring aop模式;纯pojo切面;@aspectj注解驱动的切面;@注入式aspectj切面。spring所创建的通知都是用标准的java类编写的,而且定义通知所应用的切点通常会使用注解或在spring配置文件里采用xml来编写。

spring只支持方法级别的连接点。

在spring aop中,要使用aspectj的切点表达式语言来定义切点,关于spring aop的aspectj切点,最重要的一点就是spring仅支持aspectj切点指示器的一个子集:

1.arg() 限制连接点匹配参数为指定类型的执行方法;
2.@args() 限制连接点匹配参数由指定注解标注的执行方法;
3.execution() 用于匹配是连接点的执行方法;
4.this() 限制连接点匹配aop代理的bean引用为指定类型的类
5.target 限制连接点匹配目标对象为指定类型的类
6.@target() 限制连接点匹配特定的执行对象,这些对象对应的类要具有指定类型的注解
7.within() 限制连接点匹配指定的类型
8.@within() 限制连接点匹配特定注解所标注的类型
9.@annotation 限定匹配带有指定注解的连接点

spring 注解创建切面

目标对象:

package concert;
public interface performance{
  public void perform();
}

切面对象:

package concert;

@aspect//表示audience的实例是一个切面
public class audience{
  @before("execution(**concert.performance.perform(..))")
  public void silencecellphones(){
  //在perfrom方法执行之前
  }
  @before("execution(**concert.performance.perform(..))")
  public void takeseats(){
  //在perfrom方法执行之前
  }
  @afterreturning("execution(**concert.performance.perform(..))")
  public void silencecellphones(){
  //在perfrom方法执行之后
  }
  @afterthrowing("execution(**concert.performance.perform(..))")
  public void silencecellphones(){
  //在perfrom方法抛出异常之后
  }
}

上面的类中切点表达式execution(**concert.performance.perform(..))多次出现,我们也可以通过@pointcut注解避免每次都写很长的切点表但是如下所示:

@aspect//表示audience的实例是一个切面
public class audience{
  @pointcut("execution(**concert.performance.perform(..))")
  public void performance(){}
  @before("performance()")
  public void silencecellphones(){
  //在perfrom方法执行之前
  }
  @before("performance()")
  public void takeseats(){
  //在perfrom方法执行之前
  }
  @afterreturning("performance()")
  public void silencecellphones(){
  //在perfrom方法执行之后
  }
  @afterthrowing("performance()")
  public void silencecellphones(){
  //在perfrom方法抛出异常之后
  }
}

接下来需要在配置文件中配置切面如下所示:

@configuration
@enableaspectjautoproxy//启动aspectj自动代理
@componentscan
public class concertconfig{
}
//或者在配置文件中配置中添加
<aop:aspectj-autoproxy />

表示启动切面代理

环绕通知:

@aspect//表示audience的实例是一个切面
public class audience{
  @pointcut("execution(**concert.performance.perform(..))")
  public void performance(){}
  
  @before("performance()")
  public void watchperformance(proceedingjoinpoint jp){
    //在方法之前执行
    system.out.println(" beform the method is invoked");
    jp.proceed()//控制权交给目标方法
    //在方法之后执行
    system.out.println(" after the method is invoked");
  }
  
}

处理通知中的参数

public class audience{
  @pointcut("execution(**concert.performance.perform(int))&&args(tracknumber)")
  public void performance(){}
  
  @before("performance(tracknumber)")
  public void watchperformance(int tracknumber){
    //截获传递给目标方法的参数并传递给切面中处理方法
    system.out.println(tracknumber);
  }
  
}

xml中声明切面

spring aop提供的xml配置元素:

1.<aop:advisor> 定义aop通知;
2.<aop:after> 后置通知;
3.<aop:after-returning> 返回通知
4.<aop:around> 环绕通知
5.<aop:aspect> 定义一个切面
6.<aop:aspectj-autoproxy> 启用切面注解驱动
7.<aop:before> 前置通知
8.<aop:config> 顶层的aop配置元素;
9.<aop:pointcut>:定义个切点

<aop:config>
  <aop:aspect ref="audience">
    <aop:before
     pointcut="execution(**concert.performance.perform())" method="silencecellphones"/>
     <aop:before
     pointcut="execution(**concert.performance.perform())" method="takeseats"/>
     <aop:after-returning
     pointcut="execution(**concert.performance.perform())" method="applause"/>
     <aop:after-throwing
     pointcut="execution(**concert.performance.perform())" method="demandrefund"/>

  </aop:aspect>
</aop config>

定义切点:

<aop:config>
  <aop:aspect ref="audience">
    <aop:pointcut id="performance" expression="execution(**concert.performance.perform())">
    <aop:before
     pointcut-ref="performance" method="silencecellphones"/>
     <aop:before
     pointcut="performance" method="takeseats"/>
     <aop:after-returning
     pointcut="performance" method="applause"/>
     <aop:after-throwing
     pointcut="performance" method="demandrefund"/>

  </aop:aspect>
</aop config>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。