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实战》