AOP(Aspect-Oriented Programming)即面向切面编程,它与OOP(Object-Oriented Programming, 面向对象编程)相辅相成,提供了与OOP不同的抽象软件结构的视角。在OOP中,以类作为程序的基本单元,而AOP中的基本单元是Aspect(切面)。Struts2的拦截器设计就是基于 AOP的思想,是个比较经典的应用。
在业务处理代码中通常有日志记录、性能统计、 安全控制、 事务处理、 异常处理等操作。尽管使用 OOP 可以通过封装或继承的方式达到代码的重用,但仍然有同样的代码分散在各个方法中。因此,采用OOP处理日志记录等操作不仅增加了开发者的工作量,而且提高了升级维护的困难。为了解决此类问题,AOP思想应运而生。AOP采取横向抽取机制, 即将分散在各个方法中的重复代码提取出来,然后在程序编译或运行阶段将这些抽取出来的代码应用到需要执行的地方。这种横向抽取机制采用传统的OOP是无法办到的,因为OOP实现的是父子关系的纵向重用。但是AOP不是OOP的替代品,而是 OOP的补充,它们相辅相成。
切面(Aspect)是指封装横切到系统功能(例如 事务 处理)的类。
连接点(Joinpoint)是指程序运行中的一些时间点,例如方法的调用或异常的抛出。
在Spring AOP中,所有的方法执行都是连接点
切入点(Pointcut)是指需要处理的连接点。在 Spring AOP中,所有的方法执行都是连接点,而切入点是一个描述信息,它修饰的是连接点,通过切入点确定哪些连接点需要被处理。切面、连接点和切入点的关系如图4.2所示。
通知(Advice)是由切面添加到特定的连接点(满足切入点规则)的 一段代码,即在定义好 的切入点处所要执行的程序代码,可以将其理解为切面开启后切面的方法,因此通知是切面的具体实现。
引入(Introduction)允许在现有的实现类中添加自定义的方法和属性。
目标对象(TargetObject)是指所有被通知的对象。如果AOP框架使用运行时代理的方式(动态的AOP)来实现切面,那么通知对象总是一个代理对象。
代理(Proxy)是通知应用到目标对象之后被动态创建的对象。
织入(Weaving)是将切面代码插入到目标对象上,从而生成代理对象的过程。根据不同的实现技术,AOP织入有3 种方式:编译期织入,需要有特殊的 Java 编译器;类装 载期织入, 需要有特殊的类装载器;动态代理织入,在运行期为目标类添加通知生成子类 的方式。 SpringAOP框架默认采用动态代理织入,而 AspectJ(基于 Java语言的 AOP框架)
在Java中存在多种动态代理技术,例如JDK、CGLIB、Javassist、ASM,其中最常用的动态代理技术是JDK和CGLIB,目前,在Spring AOP中常用JDK和CGLIB两种动态代理技术
JDK动态代理是java.lang.reflect.*包提供的方式,它必须借助一个接口才能产生代理对象。因此,对于使用业务接口的类,Spring默认使用JDK动态代理实现AOP。下面通过一个实例演示如何使用 JDK动态代理实现 SpringAOP,具体步骤如下:
JDK动态代理必须提供接口才能使用,对于没有提供接 口的类,只能采用CGLIB动态代理。
CGLIB(Code Generation Library)是一个高性能开源的代码生成包,采用非常底层的字节码技术,对指定的 目标类生成一个子类,并对子类进行增强。 在 SpringCore包中己经集成了CGLIB所需要的JAR包,不需要另外导入JAR包。 下面通过一个实例演示 CGLIB 动态代理的实现过程, 具体步骤如下:
在Spring中默认使用JDK动态代理实现AOP编程,使用org.springframework.aop.framework.ProxyFactoryBean创建代理是Spring AOP实现的最基本的方式。
ProxyFactoryBean是 org.springframework.beans.factory.FactoryBean接口的实现类,FactoryBean负责实例化一个Bean实例, ProxyFactoryBean负责为其他Bean实例创建代理实例。
AspectJ是一个基于 Java语言的 AOP框架。从 Spring2.0 以后引入了 AspectJ的支持。 对于目前的 Spring框架,建议开发者使用 AspectJ实现 SpringAOP。使用 AspectJ实现 Spring AOP 的方式有两种, 一是基于 XML 配置开发 AspectJ, 二是基于注解开发 AspectJ。本节 讲解基于 XML 配置开发 AspectJ 的相关知识,而基于注解开发 AspectJ 的相关知识将在 4.5 节讲解。
基于XML配置开发AspectJ是指通过XML配置文件定义切面、切入点、以及通知,所有这些定义都在aop:config元素内。
配置JDBCTemplate
在代码中显式调用beginTransaction、commit、rollback等事务处理相关的方法,这就是编程式事务管理,当只有少数事务操作时,编程式事务管理才比较合适
基于底层API的编程式事务管理就是根据PlatformTransactionManager、TransactionDefinition、TransactionStatus几个核心接口,通过编程的方式来进行事务处理。
或者基于TransactionTemplate进行编程式事务管理