Spring AOP
AOP介绍
术语
AOP
AOP为Aspect Oriented Programming的缩写,意为:面向切面编程(也叫面向方面),可以通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种技术。AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,AOP可以说也是这种目标的一种实现
切面(Aspect)
切面是你要实现的交叉功能。它是应用系统模块化的一个切面或领域。切面的最常见(虽然简单)例子是日志记录。日志记录在系统中到处需要用到,利用继承来重用日志模块不适合。然而,你可以创建一个日志记录切面,并且使用AOP在系统中应用。
连接点(Joinpoint)
连接点是应用程序执行过程中插入切面的地点。这个地点可以是方法调用,异常抛出,或者甚至是要修改的字段。切面代码在这些地方插入到你的应用流程中,添加新的行为。
通知(Advice)
通知切面的实际实现。它通知应用系统新的行为。在日志例子中,日志通知包含了实现实际日志功能的代码,如向日志文件写日志。通知在连接点插入到应用系统中。
切入点(Pointcut)
切入点定义了通知应该应用在哪些连接点。通知可以应用到AOP框架支持的任何连接点。当然,你并不希望把所有切面应用到所有可能的连接点上。切入点让你指定通知应用到什么地方。通常通过指定类名和方法名,或者匹配类名和方法名式样的正则表达式来指定切入点。一些AOP框架允许动态创建切入点,在运行时根据条件决定是否应用切面,如方法参数值。
引入(Introduction)
引入允许你为已存在类添加新方法和属性。例如,你可以创建一个稽查通知来记录对象的最后修改时间。只要用一个方法setLastMofified(Date)以及一个保存这个状态的变量。可以在不改变已存在类的情况下将这个引入,给他们新的行为和状态。
目标对象(Target)
目标对象是被通知对象。它既可以是你编写的类也可以是你要添加制定行为的第三方类。如果没有AOP,这个类就必须要包含它的主要逻辑以及其他交叉业务逻辑。有了AOP,目标对象就可以全身心地关注主要业务,忘记应用其上的通知。
代理(Proxy)
代理是将通知应用到目标对象后创建的对象。对于客户对象来说,目标对象(应用AOP之前的对象)和代理对象(应用AOP之后的对象)是一样的。也就是,应用系统的其他部分不用为了支持代理对象而改变。
织入 (Weaving)
织入是将切面应用到目标对象从而创建一个新的代理对象的过程。切面在指定接入点被织入到目标对象中。织入发生在目标对象生命周期的多个点上
编译期:切面在目标对象编译时织入.这需要一个特殊的编译器.
类装载期:切面在目标对象被载入JVM时织入.这需要一个特殊的类载入器.它在类载入到应用系统之前增强目标对象的字节码
运行期:切面在应用系统运行时织入.
Spring的AOP实现
spring实现AOP关键点:使用Java编写Spring通知
spring运行是通知对象
代理Bean只有在第一次被应用系统需要的时候才被创建,如果使用的是ApplicationContext,代理对象在BeanFactory载入所有Bean的时候创建,所以使用SpringAOP不需要特殊的编译器。
Spring有2种代理创建方式
1、如果目标对象实现了一个或多个接口暴露的方法,Spring将使用JDK的java.lang.reflect.Proxy类创建代理,这个类让Spring动态产生一个新的类,它实现了所有的接口,织入了通知,并且代理对目标对象的所有请求。
2、如果目标对象没有实现任何接口,Spring使用CGLIB库生成目标对象的子类,在创建子类的时候,Spring将通过织入,并且将对目标对象的调用委托给这个子类,
注意:
1)对接口创建代理优于对类创建代理,因为这样会产生更加松耦合的系统,对类生成代理是让遗留系统或者无法实现接口的第三方类库同样的可以得到通知,这 种方式是备用方案2)标记为final的方法不能被通知,Spring是为目标类产生子类,任何需要被通知的方法都会被复写,将通知织入,final方 法是不可能做到的
Spring 实现了AOP联盟接口
AOP联盟的目标是标准化AOPjava接口,使各种JavaAop的实现可以互通。
Spring只支持方法连接点
不同的AOP实现具有多种连接点模型。由于Spring是基于动态代理的,他只支持方法连接点。这一点于其它一些AOP框架不同,比如AspectJ和Jboss,它们提供不仅提供了方法切点,还具有字段和构造器连接点。Spring缺少字段切入点,不能让我们创建很细致的通知,比如截取对象字段的更新。而没有构造器切点,我们就不能在Bean被实例化时应用通知。
然而。由于Spring着重在于提供一个实现J2EE服务框架,而方法截取可以满足绝大部分需求。如果需要方法截取之外的功能,可以利用AspectJ来补充Spring的AOP(来着Spring in Acton)第二版
Spring的观点是属性拦截破坏了封装,面向对象的基本概念是对象自己处理工作,其他对象只能通过方法调用得到处理结果(第二版已经删除这个句了)