AOP简介
在这里向说一件之前生活中比较常见的事情就是抄电表,电表在我们日常的生活中很常见,只有我们用电它就在记录我们用的电量,一段时间后就会有人来抄电表。在这个中电表只记录我们的用电量,而不用管电量的统计等。而软件系统中的一些功能就像电表一样,这些功能需要用到应用程序的多个地方,但是我们又不希望在每个点都明确的调用她们。比如我们常用的日志、安全和事务管理。如果我们让应用对象关注自己的业务逻辑,其他方面的都交给其他的对象来处理,这些会显得代码更加的精简、清晰。
在软件中这些散布于应用中多处的功能被称为横切关注点(日志、安全等)(cross-cutting concern)。通常来说,这些横切的关注点是和应用的业务逻辑分离的(当时在很多地方是嵌入在应用的业务逻辑中,而AOP做的就是把这些关注点和业务逻辑分开,那么AOP到底是什么?
AOP是Aspect Oriented Programing的简称,被译为“面向切面的编程”。按照应用程序重构的思想,如果多个类中出现了相同的代码,那么就应该考虑将这些相同的代码抽象出来定义成一个父类或者使用委托,但是如果在整个的应用程序中都是用相同的一个或者几个基类,往往会导致应用程序变得更加复杂。而面向切面可以替代继承和委托,而且在很多的场合下可以使程序变得更加的简洁。
AOP中的一些术语
和大多数的技术一样,AOP也有自己的一些术语。这些术语用于描述一个切面。
- 连接点(Joinpoint)
连接点是程序执行的某个特定的位置,如类的初始化前、类的初始化后、类的某个方法调用前,每个方法调用后、方法抛出以上后。一个类或者一段程序代码拥有一些具有边界性质的特殊点。
- 切点(Pointcut)
每个程序都拥有多个连接点,如一个拥有两个方法的类,这两个方法都是连接点,即连接点是程序类中客观存在的事物。简单的来说也就是连接点都是可以切入的点,每一次我们可以选择一些特点的连接点切入,而我们所选的这些连接点就是切点。可以通过数据库的查询来理解切点和连接点之间的关系:连接点相当于数据库中的记录(也就是所有能够连接的点),而切点相当与查询条件。切点和连接点不是一一对应的关系,一个切点可以匹配多个连接点。
- 增强(Advice)
增强是织入连接点上的一段程序。向记录日志等,我们可以在应用程序逻辑方法完成后,去增强这个方法,也就是选择这个类的这个方法作为切点,去织入一段程序记录日志。相当于动态的封装这个方法。
- 目标对象
增强逻辑的织入目标类。如果没有AOP,那么目标业务类需要自己实现所有的逻辑。
- 引介(Introduction)
引介是一种特殊的增强,她为类添加一些属性方法。这样,及时一个业务类原本没有实现某个接口,通过AOP的引介功能,也可以动态地为该业务类添加接口实现逻辑。
- 织入(Weaving)
织入是将增强添加到目标类的具体连接点上的过程。从名字上也可以看出,AOP就像织布机一样,将目标类,增强编制在一起。根据不同的实现技术,AOP有三种织入方式。
(1)编译期织入:切面在目标类编译时被织入,这要求需用使用特殊的Java编译器。
(2)类装载期被织入:切面在目标类加载到JVM时被织入,这种方式需要特殊的类加载器(ClassLoader),它可以在目标类引入应用之前增强该目标类的字节码。
(3)动态代理织入,在运行期为目标类添加增强生成子类的方式。
Spring采用动态代理织入,AspectJ采用编译器织入和类装载期织入。
- 代理(Proxy)
一个类别AOP增强之后,就会产生一个新的类,他是如何了袁磊和增强逻辑的代理类。根据不同的代理方式,代理类既可以和原类具有相同接口的类,也可能就是原来的类,所以可以采用和调用原类相同的方式调用代理类。
- 切面(Aspect)
切面有切点和增强(引介)组成,它既包括了横切逻辑的定义,也包括连接点的定义。
AOP的实现者
- AspectJ
AspectJ是一个面向切面的框架,2001年有Xerox PARC的AOP小组发布,它扩展了Java语言。AspectJ定义了AOP语法,所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件。
- AspectWerkz
AspectWerkz是基于Java的简单、动态、轻量级的AOP框架,该框架发布于2002年,有BEA Systems提供支持。它支持运行期或类装载期织入横切代码,它也有一个特殊的类装载器。现在AspectJ和AspectWerkz项目已经合并,它们合作的第一个发布版是AspectJ 5:扩展AspectJ语言,以注解的方式支持类似AspectJ的代码风格。
- Jboss AOP
JBoss AOP于2004年作为JBoss应用服务器框架的扩展功能发布。
- Spring AOP
Spring AOP使用纯Java实现,他不需要专门的编译过程,也不需要特殊的类装载器,他在运行期通过代理方式向目标类中织入增强代码。