Spring学习笔记(六、Spring AOP基本概念)

上一篇:Spring学习笔记(五、Bean装配(下))

一、AOP概念

1. 什么是AOP?

  • AOP:Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。
  • 主要功能是:日志记录、性能统计、安全控制、事务处理、异常处理等等。

2. AOP实现方式

  • 预编译
    -AspectJ
  • 运行期动态代理(JDK动态代理,CGLib动态代理)
    -Spring AOP、Jboss AOP

3. AOP相关概念

| 名称| 说明 |
| ------------- |:-------------:| -----:|
| 切面(Aspect) |一个关注点的模块化,这个关注点可能会横切多个对象 |
| 连接点(Joinpoint) |程序执行过程中的某个特定的点|
| 通知(Advice) |在切面的某个特定的连接点上执行的动作|
| 切入点(PointCut) |匹配连接点的断言,在AOP中通知和一个切入点表达式关联|
| 引入(Introduction) |在不修改类代码的前提下,为类添加新的方法和属性|
| 目标对象(Target Object) |被一个或者多个切面所通知的对象|
| AOP代理(AOP proxy) |AOP框架创建的对象,用来实现切面契约(aspect contract)(包括通知方法执行等功能)|
| 织入(Weaving) |把切面连接到其他的应用程序类型或者对象上,并创建一个被通知的对象,分为:编译时织入、类加载时织入、执行时织入|

4. Advice的类型

| 名称| 说明 |
| ------------- |:-------------:| -----:|
| 前置通知(Before advice)|在某连接点(join point)之前执行的通知,但不能阻止连接点前的执行(除非它抛出一个异常】)|
| 返回后通知(After returning advice)|在某连接点(join point)正常完成后执行的通知|
| 抛出异常后通知(After throwing advice)|在方法抛出异常退出时执行的通知|
| 后通知(After(finally) advice)|当某连接点退出的时候执行的通知(不论时正常返回还是异常退出)|
| 环绕通知(Around advice)|包围一个连接点(join point)的通知|

5. Spring框架中AOP的用途

  • 提供了声明式的企业服务,特别是EJB替代服务的声明
  • 允许用户定制自己的方面,以完成AOP和OOP的互补使用

6. Spring的AOP实现

  • 纯java实现,无需特殊的编译过程,不需要控制类加载器层次
  • 目前只支持方法执行连接点(通过Spring Bean的方法执行)
  • 不是为了提供最完整的AOP实现(尽管它非常强大);而是侧重于提供一种AOP实现和Spring IoC容器之间的整合,用于帮助企业应用中的常见问题。
  • Spring AOP不会与AspectJ竞争,从而提供综合全面的AOP解决方案。

7. 有接口和无接口的Spring AOP实现区别

  • 业务对象有接口, Spring AOP 默认使用标准的JavaSE动态代理作为AOP代理,这使得任何接口(或者接口集)都可以被代理。
  • 业务对象没有接口,可以使用Spring AOP中CGLIB代理

二、配置切面aspect

1. 基于Schema-based的AOP实现

  • Spring所有切面和通知器都必须放在<aop:config>内(可以配置包含多个<aop:config>元素),每一个<aop:config>可以包含pointcut、adviser和aspect元素(他们必须按照这个顺序进行声明)。
  • <aop:config>风格的配置大量使用了Spring的自动代理机制。

示例

  1. 创建切面类MoocAspect :
package test14;
/**
 * Created by amber on 2017/6/18.
 * 切面类
 */
public class MoocAspect {
}
  1. 创建目标类
package test14;
/**
 * Created by amber on 2017/6/18.
 */
public class AspectBiz {
}
  1. ApplicationContext中配置:
    在<beans>中声明命名空间
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-context.xsd"
 <aop:config>
        <aop:aspect id="myAspect" ref="moocAspect"></aop:aspect>
    </aop:config>
    <bean id="aspectBiz" class="test14.AspectBiz"></bean>
    <bean id="moocAspect" class="test14.MoocAspect"/>

2、配置切入点Pointcut

  • 在AOP中通知(Advice)和一个切入点表达式关联。
Paste_Image.png
execution(public **(..)) 
切入点为执行所有public方法时
execution(* set*(..)) 
切入点为执行所有set开始的方法时
execution(* com.xyz.service.AccountService.*(..)) 
切入点为执行AccountService类中所有方法时
execution(* com.xyz.service..(..)) 
切入点为执行 com.xyz.service包下所有方法时
execution(* com.xyz.service...(..)) 
切入点为执行 com.xyz.service包下及其子包下所有方法时
within(com.xyz.service.*) (仅SpringAOP)
within(com.xyz.service..*) (仅SpringAOP)
within用于匹配指定类型内的方法执行
this(com.xyz.service.AccountService) (仅SpringAOP)
this用于匹配当前AOP代理对象类型的执行方法
target(com.xyz.service.AccountService) (仅SpringAOP)
target用于匹配当前目标对象类型的执行方法
args(java.io.Serializable) (仅SpringAOP)
args用于匹配当前执行的方法传入的参数为指定类型的执行方法
@target(org.springframework.transacation.annotation.Transcactional) (仅SpringAOP)
@within(org.springframework.transacation.annotation.Transcactional) (仅SpringAOP)
@annotation(org.springframework.transacation.annotation.Transcactional) (仅SpringAOP)
按照注解类型匹配
@args(com.xyz.security.Classified)(仅SpringAOP)
bean(tradeService)(仅SpringAOP)
bean(*Service)(仅SpringAOP)

配置切入点:

   <aop:config>
        <aop:aspect id="myAspect" ref="moocAspect">
            <aop:pointcut id="moocPointCut" expression="execution(* test14.*Biz(..))"/>
        </aop:aspect>
    </aop:config>

三、advice

** Before Advice(前置通知)**
在写测试之前呢,要现在下载好需要的jar包。

        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.8.10</version>
        </dependency>
  1. 创建了业务类AspectBiz:
package test14;
/**
 * Created by amber on 2017/6/18.
 * 业务类
 */
public class AspectBiz {
   public void biz(){
       System.out.println("AspectBiz biz.");
   }
}
  1. 创建切面类MoocAspect:
package test14;
/**
 * Created by amber on 2017/6/18.
 * 切面类
 */
public class MoocAspect {
    public void before(){
        System.out.println("MoocAspect 前置通知,before");
    }
}
  1. applicationContext中:
<bean id="aspectBiz" class="test14.AspectBiz"></bean>
 <bean id="moocAspect" class="test14.MoocAspect"/>
    <aop:config>
        <aop:aspect id="moocAspectAop" ref="moocAspect">
            <aop:pointcut id="moocPointCut" expression="execution(* test14.*Biz.*(..))"/>
            <aop:before method="before" pointcut-ref="moocPointCut"/>
        </aop:aspect>
    </aop:config>
  1. 测试类:
  @Test
    public void test14() {
        AspectBiz aspectBiz=super.getBean("aspectBiz");
        aspectBiz.biz();
    }
  1. 结果:


    Paste_Image.png

** After returning advice(返回后通知)**

  1. 更新MoocAspect类:
package test14;
/**
 * Created by amber on 2017/6/18.
 * 切面类
 */
public class MoocAspect {
    public void before(){
        System.out.println("MoocAspect 前置通知,before");
    }
    public void afterReturning(){
        System.out.println("MoocAspect 返回后通知,afterReturning");
    }
}
  1. 更新applicationContext:
 <aop:config>
        <aop:aspect id="moocAspectAop" ref="moocAspect">
            <aop:pointcut id="moocPointCut" expression="execution(* test14.*Biz.*(..))"/>
            <aop:before method="before" pointcut-ref="moocPointCut"/>
            <aop:after-returning method="afterReturning" pointcut-ref="moocPointCut"/>
        </aop:aspect>
    </aop:config>
  1. 结果:


    Paste_Image.png

** After throwing advice(抛出异常后通知)**
注:可使用throwing属性指定可被传递的异常参数名称

  1. 修改业务类AspectBiz:
package test14;
/**
 * Created by amber on 2017/6/18.
 * 业务类
 */
public class AspectBiz {
   public void biz(){
       System.out.println("AspectBiz biz.");
      throw  new RuntimeException();
   }
}
  1. 更新切面类MoocAspect :
package test14;
/**
 * Created by amber on 2017/6/18.
 * 切面类
 */
public class MoocAspect {
    public void before(){
        System.out.println("MoocAspect 前置通知,before");
    }
    public void afterReturning(){
        System.out.println("MoocAspect 返回后通知,afterReturning");
    }
    public void afterThrowing(){
        System.out.println("MoocAspect 抛出异常后通知,afterThrowing");
    }
}
  1. 更新applicationContext:
 <aop:config>
        <aop:aspect id="moocAspectAop" ref="moocAspect">
            <aop:pointcut id="moocPointCut" expression="execution(* test14.*Biz.*(..))"/>
            <aop:before method="before" pointcut-ref="moocPointCut"/>
            <aop:after-returning method="afterReturning" pointcut-ref="moocPointCut"/>
            <aop:after-throwing method="afterThrowing" pointcut-ref="moocPointCut"/>
        </aop:aspect>
    </aop:config>
  1. 结果:


    Paste_Image.png

** After(finally) advice(后通知)**

  1. 更新切面类MoocAspect:
package test14;
/**
 * Created by amber on 2017/6/18.
 * 切面类
 */
public class MoocAspect {
    public void before(){
        System.out.println("MoocAspect 前置通知,before");
    }
    public void afterReturning(){
        System.out.println("MoocAspect 返回后通知,afterReturning");
    }
    public void afterThrowing(){
        System.out.println("MoocAspect 抛出异常后通知,afterThrowing");
    }
    public void after(){
        System.out.println("MoocAspect 后通知,after");
    }
}
  1. 更新applicationContext:
  <aop:config>
        <aop:aspect id="moocAspectAop" ref="moocAspect">
            <aop:pointcut id="moocPointCut" expression="execution(* test14.*Biz.*(..))"/>
            <aop:before method="before" pointcut-ref="moocPointCut"/>
            <aop:after-returning method="afterReturning" pointcut-ref="moocPointCut"/>
            <aop:after-throwing method="afterThrowing" pointcut-ref="moocPointCut"/>
            <aop:after method="after" pointcut-ref="moocPointCut"/>
        </aop:aspect>
    </aop:config>
  1. 结果:
Paste_Image.png

** Around advice(环绕通知)**
注:通知方法的必须有参数,且第一个参数必须是ProceedingJoinPoint类型。

  1. 修改AspectBiz 业务类,讲抛出异常代码删掉。
package test14;
/**
 * Created by amber on 2017/6/18.
 * 业务类
 */
public class AspectBiz {
   public void biz(){
       System.out.println("AspectBiz biz.");
   }
}
  1. 更新MoocAspect类:
package test14;
import org.aspectj.lang.ProceedingJoinPoint;
/**
 * Created by amber on 2017/6/18.
 * 切面类
 */
public class MoocAspect {
    public void before(){
        System.out.println("MoocAspect 前置通知,before");
    }
    public void afterReturning(){
        System.out.println("MoocAspect 返回后通知,afterReturning");
    }
    public void afterThrowing(){
        System.out.println("MoocAspect 抛出异常后通知,afterThrowing");
    }
    public void after(){
        System.out.println("MoocAspect 后通知,after");
    }
    public Object around(ProceedingJoinPoint pjp){
        Object obj= null;
        try {
            System.out.println("MoocAspect 环绕通知,around1");
            obj = pjp.proceed();
            System.out.println("MoocAspect 环绕通知,around2");
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        return obj;
    }
}
  1. applicationContext:
  <aop:config>
        <aop:aspect id="moocAspectAop" ref="moocAspect">
            <aop:pointcut id="moocPointCut" expression="execution(* test14.*Biz.*(..))"/>
            <aop:before method="before" pointcut-ref="moocPointCut"/>
            <aop:after-returning method="afterReturning" pointcut-ref="moocPointCut"/>
            <aop:after-throwing method="afterThrowing" pointcut-ref="moocPointCut"/>
            <aop:after method="after" pointcut-ref="moocPointCut"/>
            <aop:around method="around" pointcut-ref="moocPointCut"/>
        </aop:aspect>
    </aop:config>
  1. 结果:
Paste_Image.png

** Advice parameters(通知参数)**

  1. 更新AspectBiz 业务类,增加一个带参的方法,init。
package test14;
import test12.StringStore;
/**
 * Created by amber on 2017/6/18.
 * 业务类
 */
public class AspectBiz {
   public void biz(){
       System.out.println("AspectBiz biz.");
   }
    public void init(String bizName, int times){
        System.out.println("AspectBiz init: "+bizName+" "+times);
    }
}
  1. 更新MoocAspect 切面类,增加带参数环绕通知方法aroundInit
package test14;
import org.aspectj.lang.ProceedingJoinPoint;
import test12.StringStore;
/**
 * Created by amber on 2017/6/18.
 * 切面类
 */
public class MoocAspect {
    public void before(){
        System.out.println("MoocAspect 前置通知,before");
    }
    public void afterReturning(){
        System.out.println("MoocAspect 返回后通知,afterReturning");
    }
    public void afterThrowing(){
        System.out.println("MoocAspect 抛出异常后通知,afterThrowing");
    }
    public void after(){
        System.out.println("MoocAspect 后通知,after");
    }
    public Object around(ProceedingJoinPoint pjp){
        Object obj= null;
        try {
            obj = pjp.proceed();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }

        return obj;
    }
    public Object aroundInit(ProceedingJoinPoint pjp, String bizName,int times){
        System.out.println("MoocAspect 环绕通知,bizName: "+bizName+" times: "+times);
        Object obj= null;
        try {
            System.out.println("MoocAspect 环绕通知,around1");
            obj = pjp.proceed();
            System.out.println("MoocAspect 环绕通知,around2");
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
        return obj;
    }
}
  1. appclicationContext:
  <aop:config>
        <aop:aspect id="moocAspectAop" ref="moocAspect">
            <aop:pointcut id="moocPointCut" expression="execution(* test14.*Biz.*(..))"/>
            <aop:before method="before" pointcut-ref="moocPointCut"/>
            <aop:after-returning method="afterReturning" pointcut-ref="moocPointCut"/>
            <aop:after-throwing method="afterThrowing" pointcut-ref="moocPointCut"/>
            <aop:after method="after" pointcut-ref="moocPointCut"/>
            <aop:around method="around" pointcut-ref="moocPointCut"/>
            <aop:around method="aroundInit" pointcut="execution(* test14.AspectBiz.init(String,int))
            and args(bizName,times)"/>
        </aop:aspect>
    </aop:config>
  1. 测试类:
  @Test
    public void test14() {
        AspectBiz aspectBiz=super.getBean("aspectBiz");
        aspectBiz.init("advice parameters",1);
    }
  1. 结果:


    Paste_Image.png

四、Introductions

  • Introductions允许一个切面声明一个实现指定接口的通知对象,并且提供了一个接口实现类来代表这些对象。
  • 由<aop:aspect>中的<aop:declare-parents>元素声明,该元素用于声明所匹配的类型拥有一个新的parent(因此得名)
  1. 新增接口Fit:
package test15;
/**
 * Created by amber on 2017/6/18.
 */
public interface Fit {
    void filter();
}
  1. 新增实现类FitImpl :
package test15;
/**
 * Created by amber on 2017/6/18.
 */
public class FitImpl implements Fit {
    public void filter() {
        System.out.print("FitImpl filter");
    }
}
  1. 更新applicationContext:
  <bean id="aspectBiz" class="test14.AspectBiz"></bean>
   <bean id="moocAspect" class="test14.MoocAspect"/>
    <aop:config>
        <aop:aspect id="moocAspectAop" ref="moocAspect">
            <aop:declare-parents types-matching="test14.*+" 
                                 implement-interface="test15.Fit" 
                                 default-impl="test15.FitImpl"/>
        </aop:aspect>
    </aop:config>
在<aop:declare-parents>标签中,types-matching是test14包下的所有类,implement-interface是Fit接口。
default-impl是默认的实现类FitImpl。
在测试类中 Fit fit=super.getBean("aspectBiz");
意思是给AspectBiz类增加了一个父类FitImpl 。
上面“提供了一个接口实现类来代表这些对象”,即用实现Fit接口的FitImpl类代表AspectBiz对象,
  1. 测试类:
 @Test
    public void test15() {
        Fit fit=super.getBean("aspectBiz");
        fit.filter();
    }
Paste_Image.png

最后注意:所有基于配置文件的Aspect,只支持单例模式。

五、Advisors

  • Advisors就像一个小的自包含的方面,只有一个advice。
  • 切面自身通过一个bean表示,并且必须实现某个advice接口,同时,Advisor也可以很好的利用AspectJ的切入点表达式。
  • Spring通过配置文件中的<aop:advisor>元素支持advisor。实际使用中,大多数情况下它会和transactional advice配合使用。
  • 为了定义一个advisor的优先级以便让advice可以有序,可以使用order属性来定义advisor的顺序。
    示例:
    创建一个实现某个Advice的切面类:
package com.myspring.app.aop;
import java.lang.reflect.Method;
import org.aspectj.lang.JoinPoint;
import org.springframework.aop.MethodBeforeAdvice;
/**
 * 方法前置通知
 * @author Michael
 */
@Component//如果是自动装配,在定义切面的时候直接写在ref属性里就可以了
public class MyAdvice implements MethodBeforeAdvice{
    //如果使用aop:advisor配置,那么切面逻辑必须要实现advice接口才行!否则会失败!
    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
        System.out.println("前置通知");
    }
    //如果是<aop:aspect>配置,编写一般的方法就可以了,然后在切面配置中指定具体的方法名称!
    public void doBefore(JoinPoint point) throws Throwable {
    }
}

applicationContext(重点!!可以和<aop:aspect>对比着看):

    <aop:config>
        <aop:pointcut expression="(execution(* com.myspring.app.aop.TestPoint.*(..)))"  id="mypoint"/>
        <aop:advisor advice-ref="myAdvice " pointcut-ref="mypoint"/>
    </aop:config>

六、验证可以通过切面控制目标执行次数的示例:

创建ConrrentOperationExecutor 切面类。

package schema.advisors;
import org.aspectj.lang.ProceedingJoinPoint;
import org.springframework.dao.PessimisticLockingFailureException;
import test12.StringStore;
/**
 * Created by amber on 2017/6/18.
 */
public class ConrrentOperationExecutor {
    private int maxRetries;
    public void setMaxRetries(int maxRetries) {
        this.maxRetries = maxRetries;
    }
    public Object doCurrentOperation(ProceedingJoinPoint pjp)throws Throwable{
        int numAttempts=0;
        PessimisticLockingFailureException lockingFailureException;
        do{
            numAttempts++;
            System.out.println("Try times : "+numAttempts);
            try {
                return pjp.proceed();
            } catch (PessimisticLockingFailureException throwable) {
                lockingFailureException=throwable;
            }
        }while (numAttempts<this.maxRetries);
        System.out.println("Try error : "+numAttempts+"\n");
        throw lockingFailureException;
    }
}

创建InvokeService业务类:

package schema.advisors.service;
import org.springframework.dao.PessimisticLockingFailureException;
import org.springframework.stereotype.Service;
/**
 * Created by amber on 2017/6/18.
 */
@Service
public class InvokeService {
    public void invoke() {
        System.out.print("InvokeService invoke...");
    }
    public void invokeException() {
     throw new PessimisticLockingFailureException("");
    }
}

在ApplicationContext:

   <context:component-scan base-package="schema"></context:component-scan>

    <aop:config>
        <aop:aspect id="conrrentOperationRetry" ref="conrrentOperationExecutor">
            <aop:pointcut id="idempotentOperation" expression="execution(* schema.advisors.service.*.*(..))"/>
            <aop:around method="doCurrentOperation" pointcut-ref="idempotentOperation"/>
        </aop:aspect>

    </aop:config>

    <bean id="conrrentOperationExecutor" class="schema.advisors.ConrrentOperationExecutor">
        <property name="maxRetries" value="3"/>
    </bean>

测试方法:

  @Test
    public void schema() {
        InvokeService service=super.getBean("invokeService");
        service.invoke();
        System.out.println();
        service.invokeException();
    }

结果:


Paste_Image.png

下一篇:Spring学习笔记(七、Spring AOP API)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容