Spring AOP是AOP技术在Spring中的具体实现,它是Spring框架的另外一个重要基石。
Spring构建于IOC之上,和IOC浑然天成,统一于Spring容器之中。
AOP的出现并不是要取代面向对象编程OOP(Object-oriented programming),而是作为OOP的有益补充。
AOP是有特定的应用场合的,它只适合那些具有横切逻辑的应用场合,比如性能监测、访问控制、事务管理以及日志记录等。
AOP(Aspect-oriented programming)面向切面编程。
按照软件重构思想的理念,如果多个类中出现了相同的代码,则应该考虑定义一个父类,将这些相同的代码提取到父类中。 比如 Dog、Cat、Pig都有run()和eat()方法,通过引入一个包含这两个方法的抽象的Animal父类,Dog、Cat、Pig就可以通过继承Animal复用run()和eat()方法方法。
代码已托管到Github—> https://github.com/yangshangwei/SpringMaster
通过引入父类消除多个类中重复代码的方式在大多数情况下是可行的,但有的时候并非可以解决,比如:
ForumService.java
package com.xgj.aop.introduce;
public class ForumService {
private TransactionManager transManager;
private PerformanceMonitor pmonitor;
private TopicDao topicDao;
private ForumDao forumDao;
public void removeTopic(int topicId) {
// 性能监视开始
pmonitor.start();
// 事务开始
transManager.beginTransaction();
// (1)业务代码
topicDao.removeTopic(topicId);
// 事务结束
transManager.endTransaction();
// 性能监视结束
pmonitor.end();
}
public void CreateForum(Forum forum) {
pmonitor.start();
transManager.beginTransaction();
// (2)业务代码
forumDao.create(forum);
transManager.endTransaction();
pmonitor.end();
}
}
我们发现(1)(2)两处真正的业务代码被 性能监视代码 和 事务方法的所包围。 业务代码淹没在重复化非业务性的代码之中。 性能监视和事务管理这些非业务性代码葛藤缠树般包围业务代码。
如果将ForumService业务类看做一段圆木, 将removeTopic()和createForum()方法分别看成圆木的一截,会发现性能监视和事务管理的代码就像一个年轮,而业务代码就是圆木的树心,这也是横切代码概念的由来。
横切逻辑示意图:
我们无法通过抽象父类的方式消除如上重复性横切代码,因为这些横切逻辑依附在业务类方法的流程中,它们不能转移到其他地方去。
AOP独辟蹊径,通过横向抽取机制为这类无法通过纵向继承体系进行抽象的重复性代码提供了解决方案。
横向抽取:
从上图可以看出: AOP希望将这些分散在各个业务逻辑代码中的相同代码通过横向切割的方式抽取到一个独立的模块中,还业务逻辑一个清新的世界。
当然,将这些重复性的横切逻辑独立出来是容易的,但是如何将这些独立的逻辑融合到业务逻辑中以完成和原来一样的业务流程,这才是关键,这也是AOP要解决的主要问题
连接点是程序执行的某个特定位置。比如类初始化前、类初始化后、类的某个方法调用前\调用、方法抛出异常后等。
一个类或者一段程序代码拥有一些具有边界性质的特定点,这些代码中的特定点被称为“连接点”
Spring仅支持方法的连接点,即仅仅能在方法调用前、调用后、方法抛出异常时以及方法调用前后这些程序执行点织入增强。
连接点由两个信息确定:
比如Test.foo()方法执行前的连接点,执行点为Test.foo(),方位为该方法执行前的位置。 Spring使用切点对执行点进行定位,而方位则在增强类型中定义。
每个程序类都拥有多个连接点,比如拥有两个方法的类,这两个方法都是连接点。
在众多的连接点中,如何定位我们需要的连接点呢? AOP通过“切点”定位特定的连接点。
一个切点可以匹配多个连接点。
在Spring中,切点通过 org.springframework.aop.Pointcut接口类进行描述,它使用类和方法作为连接点的查询条件,Spring AOP的规则引擎负责解析切点所设定的查询条件,找到对应的连接点。
确切的说,应该是执行点而非连接点,因为连接点是方法执行前、执行后等包含方位信息的具体程序执行点, 而切点只是定位到某个方法上, 如果希望定位到具体的连接点上,还需要提供方位信息,
增强是织入目标类连接点上的一段程序代码。
在Spring中,增强除了用于描述一段程序代码外,还拥有另一个和连接点相关的信息,这便是执行点的方位。 结合执行点的方位信息和切点信息,就可以找到特定的连接。
正是因为增强既包含用于添加到目标连接点上的一段执行逻辑,又包含用于特定连接点的方位信息,所以Spring提供的增强接口都是带方位名的,比如BeforeAdvice、AfterReturningAdvice、ThrowsAdvice等。
所以只有结合切点和增强,才能确定特定的连接点并实施增强逻辑。
增强逻辑的织入目标类。
如果没有AOP,业务目标类需要自己实现所有的逻辑,如最上面的代码(性能测试和事务管理)。 在AOP的帮助下,ForumService只需要试下那些非横切逻辑的程序逻辑,而性能监视和事务管理等这些横切逻辑则可以使用AOP动态织入特定的连接点上。
引介是一种特殊的增强,它为类添加一些属性和方法。
这样,即使一个业务类原本没有实现某个接口,通过AOP的引介功能,也可以动态的为该业务类添加接口的实现逻辑,让业务类成为这个接口的实现类。
织入是将增强添加到目标的具体连接点上的过程。
AOP有三种织入方式
Spring采用动态代理织入 ,而AspectJ 采用编译期织入和装载期织入
一个类被AOP织入增强后,就产生了一个结果类,它是融合了原类和增强逻辑的代理类。
切面由切点和增强(引介)组成。它既包括横切逻辑的定义,也包括连接点的定义。
SoringAOP就是负责实施切面的框架,它将切面所定义的横切逻辑织入切面所指定的连接点中。
AOP的工作重心在于如何将增强应用于目标对象的连接点上。 主要包括两点:
AspectJ是目前最完善的AOP语言,对Java编程语言进行了扩展,定义了AOP语法,能够在编译期提供横切代码的织入。AspectJ提供了两种横切实现机制,一种称为动态横切(Dynamic Crosscuting),另一种成为静态横切(Static Crosscuting)
基于java的简单、动态和轻量级的AOP框架,支持运行期或者类装载期织入横切代码,它拥有一个特殊的类装载器。他与AspectJ项目已经合并,第一个发行版是AspectJ5。扩展AspectJ语言,以基于注解的方式支持类似AspectJ的代码风格。
JBoss是一个开源的符合J2EE规范的应用服务器,作为J2EE规范的补充,JBoss中引入了AOP框架,为普通Java提供了J2EE服务,而无需遵循EJB规范。JBoss通过类载入时,使用Javassis对字节码操作实现动态AOP框架。
Spring AOP使用纯Java实现,不需要专门的编译过程,不需要特殊的类装载器,他在运行期通过代理方式向目标类织入增强代码。Spring并不尝试提供最完整的AOP实现,主要侧重一种和Spring IoC容器整合的AOP实现,以解决企业级开发中的常见问题。