[Spring实战系列](16)面向切面编程(AOP)概述

Posted SunnyYoona

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[Spring实战系列](16)面向切面编程(AOP)概述相关的知识,希望对你有一定的参考价值。

1. 简介


在软件中,有些行为对于大多数应用都是通用的。日志,安全和事务管理的确很重要,但他们是都是应用对象主动参与的行为呢?如果让应用对象只关注自己所针对的业务领域问题,而其他方面的问题由其他应用对象来处理,这样会不会更好?

在软件开发中,分布于应用中多处的功能被称为横切关注点。通常,这些横切关注点从概念上是与应用的业务逻辑相分离的但是往往直接嵌入到应用的业务逻辑中)。将这些横切关注点与业务逻辑相分离是面向切面编程索要解决的

上图展示了一个被划分为模块的典型应用。每个模块的核心功能都是为特定业务领域提供服务,但是这些模块都需要类似的辅助功能,例如安全和事务管理。

继承与委托是最常见的实现重用通用功能的面向对象技术。但是,如果在整个应用中使用相同的基类,继承往往会导致一个脆弱的对象体系;而使用委托可能需要委托对象进行复杂的调用。切面提供了取代继承与委托的另一种选择。在使用面向切面编程时,我们让然在一个地方定义通用功能,但是我们可以通过声明的方式定义这个功能以何种方式在何处应用,而无需修改受影响的类

2. 几个AOP概念

描述AOP功能的术语并不直观,在这共同学习一下。
技术分享
技术分享
2.1 通知
在AOP术语中,切面的工作被称为通知。通知定义了切面是什么以及何时使用。除了描述切面要完成的工作,通知还解决了何时执行这个工作的问题。它该应用与某个方法被调用之前?之后?

类型 说明
Before 在方法被调用之前调用通知。
After 在方法完成之后调用通知,无论方法执行是否成功。
After-returning 在方法成功执行之后调用通知。
After-throwing
在方法抛出异常后调用通知。
Around 通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行自定义的行为。
2.2 连接点
连接点是在应用执行过程中能够插入切面的一个点这个点可以是调用方法时,抛出异常时,甚至修改一个字段时。切面代码可以利用这些点插入到应用的正常流程之中,并添加新的行为
2.3 切点
一个切面并不需要通知应用的所有连接点。切点有助于缩小切面所通知连接点的范围
2.4 切面
切面是通知和切点的结合。通知和切点共同定义了关于切面的全部内容-----它是什么,在何时和在何处完成其功能。
2.5 引入
引入允许我们向现有的类添加新方法或属性。例如,我们可以创建一个Auditable通知类,该类记录了对象最后一次修改时的状态。我们只需一种方法,setLastModified(Date),和一个实例变量来保存这个状态。然后,这个新方法和实例变量就可以被引入到现有的类中。从而可以在无需修改这些现有的类的情况下,让他们具有新的行为和状态。
2.6 织入
织入是将切面应用到目标对象来创建新的代理对象的过程。切面在指定的连接点被织入到目标对象中。在目标对象的生命周期里有多个点可以进行织入。

3. Spring对AOP的支持


并不是所有的AOP框架都是一样的,尽管有些不同,但是无论如何,创建切点来定义切面织入的连接点是AOP框架的基本功能。

现在主要的AOP框架:
  • AspecJ
  • JBoss AOP
  • Spring AOP

在这里我们主要看Spring对AOP的支持,Spring提供了4种各具特色的AOP支持:
  • 基于代理的经典AOP
  • @AspectJ注解驱动的切面
  • 纯POJO切面
  • 注入式AspectJ切面(适合Spring各版本)
3.1 Spring通知是Java编写的
Spring所创建的通知都是用标准的Java类编写的。这样的话,我们就可以使用与普通Java开发一样的集成开发环境(IDE)来开发切面。而且,定义通知所应用的切点通常在Spring配置文件里采用XML来编写的。AspectJ与之相反,虽然AspectJ现在支持基于注解的切面,但是AspectJ最初是以Java语言扩展的方式实现的。这种方式既有优点也有缺点。通过特有的AOP语言,我们可以获得更强大和细粒度的控制,以及更丰富的AOP工具集,但是需要我们学习额外的新工具和语法。
3.2 Spring在运行期通知对象
通过在代理类中包裹切面,Spring在运行期将切面织入到Spring管理的Bean中。代理类封装了目标类,并拦截被通知的方法的调用,再将调用转发给真正的目标Bean。
技术分享
技术分享
当拦截到方法调用时,在调用目标Bean方法之前,代理会执行切面逻辑。知道应用需要被代理的Bean时,Spring才创建代理对象。如果使用的是ApplicationContext,在ApplicationContext从BeanFactory中加载所有的Bean时,Spring创建被代理的对象。因为Spring运行时才创建代理对象,所以我们不需要对特殊的编译器来织入Spring AOP的切面。
3.3 Spring只支持方法连接点
因为Spring是基于动态代理的,所以Spring只支持方法连接点。这与其他一些AOP架构是不同的,例如AspectJ和JBoss,除了方法切点,它们还提供字段和构造器接入点。Spring缺少字段连接点的支持,无法创建更细粒度的通知,例如拦截对象字段的修改,而且Spring也不支持构造器连接点,我们也无法在Bean创建时应用通知。



来源于:《Spring实战》




























以上是关于[Spring实战系列](16)面向切面编程(AOP)概述的主要内容,如果未能解决你的问题,请参考以下文章

Java实战之03Spring-03Spring的核心之AOP(Aspect Oriented Programming 面向切面编程)

Spring面向切面编程AOP(around)实战

Spring系列Spring AOP面向切面编程

Spring AOP——Spring 中面向切面编程

Spring一文带你吃透AOP面向切面编程技术(上篇)

Spring实战之切面编程