Spring参考手册 AOP
[code]
概述
Spring 2.0引入了一个简单而强大的机制,使用基于模式和@AspectJ注记样式实现自定义方面。Spring当前仅支持方法执行连接点而不支持字段拦截。 Spring AOP框架的目标并不是实现完整的AOP实现,主要目的是帮助IOC容器解决一些企业应用的常见问题。Spring AOP框架通常需要和Spring IOC容器一起使用,方面使用通常的Bean定义语法。如果需要一个全面的AOP解决方案,AspectJ是更好的选择。
Spring AOP框架是基于代理的,Spring缺省使用JDK的动态代理,JDK代理可以代理任何接口。而CGLIB代理主要用来代理类,如果一个Bean没有实现一个接口,Spring将使用CGLIB进行代理。
基本概念
Aspect……横切多个对象的关注点。事务管理是J2EE应用的一个横切关注点的极好事例。
Join Point……程序执行中的一个点,例如一个方法的执行或者异常处理,Spring中连接点总是指方法执行。
Advice……方面在一个特定的连接点执行的动作,许多AOP框架,包括Spring都将advice模型为一个拦截器,基于一个连接点维持一个拦截器堆栈。
Pointcut……用来匹配连接点的语法。 Advice和一个Pointcut关联,并且运行在Pointcut匹配的连接点处(比如具有指定名称的方法)。Spring使用AspectJ的 Pointcut语法。
Introduction……针对某个类型声明额外的方法和字段,Spring AOP允许针对任何代理的对象引入新的接口。比如可以使用introduction让一个Bean实例实现IsModified接口。
Target Object……被方面拦截的对象,Spring AOP使用运行时代理实现方面拦截机制,因此这些对象都是代理对象。
Aop Proxy……AOP框架为了实现方面规约而创建的代理对象。Spring框架中,AOP代理是一个JDK动态代理或者CGLIB代理。
Weave……将方面和其他对象类型进行关联创建Advice的方式,织入可以在编译时,加载时或运行时进行,Spring AOP使用运行时织入的方式。
Advice类型
Before Advice……连接点之前执行,除非抛出异常将无法阻止执行连接点。
After Return Advice……在连接点正常返回后执行。
After throwing advice……连接点抛出异常后执行。
After (finally) advice……不管连接点正常退出还是抛出异常都会执行。
Around advice……在连接点之前和之后执行,可以在方法执行前后添加自定义逻辑,可以终止连接点的继续执行,返回值或抛出异常。
@AspectJ支持
启用AspectJ
启用@AspectJ方面时,如果Spring检测到一个实例需要被一个或多个代理拦截,Spring 将自动为这个实例生成一个代理,拦截方法执行,启用AspectJ需要添加如下配置:
xml 代码
1. <aop:aspectj-autoproxy/>
声明方面
AspectJ 对方面的支持是使用Annotation实现的,Spring将自动检测使用过@Aspect注记的Bean实例,配置AOP代理。AspectJ的方面 就是一个使用了@Aspect注记的类,和普通类一样,可以包含方法,字段,另外也可以包含pointcut,advice,introduction 等。
声明Pointcut
Pointcut 用来决定哪些方法需要应用方面,其声明包括两部分,一部分是返回值为void的方法签名,一部分是@Pointcut注记,@Pointcut注记的语法 可以参见AspectJ参考指南。下面的示例定义了名称为anyOldTransfer的Pointcut,匹配任何方法名称为transfer:
java 代码
1. @Pointcut("execution(* transfer(..))")// the pointcut expression
2. private void anyOldTransfer() {}// the pointcut signature
Spring的Pointcut支持如下形式的限定方式:
Within…..限制在指定的类型范围。
This……限制匹配的连接点所在的Bean实例的Spring AOP代理是指定的类型。
Target……限制匹配的连接点所在的Bean实例是指定的类型。
Args……限制匹配的连接点的参数是指定类型的实例。
@target……限制连接点所在的类有一个指定类型的注记。
@args……限制传递给连接点的参数的运行期类型有指定类型的注记。
@within……限制连接点在有特定注记的类型范围。
@annotation……限制连接点所在的 Bean实例的Spring AOP代理有指定的注记。
Pointcut表达式可以使用&&,||,!进行组合,也可以使用名称引用pointcut表达式,使用名称引用时,适用通常的java可见性规则,下面是几个示例:
java 代码
1. @Pointcut("execution(public * *(..))")
2. private void anyPublicOperation() {}
3. @Pointcut("within(com.xyz.someapp.trading..*")
4. private void inTrading() {}
5. @Pointcut("anyPublicOperation() && inTrading()")
6. private void tradingOperation() {}
声明Advice
Advice和一个Pointcut表达式关联,运行在Pointcut表达式匹配的方法执行前后。
针对几种不同类型的Advice,使用不同的注记来声明,比如使用@Before标记声明 before advice。
任何Advice方法可以声明一个org.aspectj.lang.JoinPoint 类型的参数,从而获取拦截方法的信息。
可以在声明Advice时,将拦截方法的参数传递进来,下面的示例想拦截第一个参数为 Account的方法,而且可以在advice方法内访问account对象。
java 代码
1. @Before("com.xyz.myapp.SystemArchitecture.dataAccessOperation() &&" +
2. "args(account,..)")
3. public void validateAccount(Account account) {
4. // ...
5. }
代理对象,目标对象和注记也可以相同的方式访问,下面的示例拦截使用@Auditable
的方法,而且可以在advice方法内访问auditable对象。
java 代码
1. @Before("com.xyz.lib.Pointcuts.anyPublicMethod() && " +
2. "@annotation(auditable)")
3. public void audit(Auditable auditable) {
4. AuditCode code = auditable.value();
5. // ...
6. }
Introduction
Introduction使用@DeclareParents注记声明,这个注记用来声明匹配pointcut的类将实现一个新的接口。下面的示例声明所有的服务接口将实现UsageTracked接口:
java 代码
1. @Aspect
2. public class UsageTracking {
3. @DeclareParents(value="com.xzy.myapp.service.*+",
4. defaultImpl=DefaultUsageTracked.class)
5. public static UsageTracked mixin;
6. @Before("com.xyz.myapp.SystemArchitecture.businessService() &&" +
7. "this(usageTracked)")
8. public void recordUsage(UsageTracked usageTracked) {
9. usageTracked.incrementUseCount();
10. }
11. }
Aspect实例模型
缺省情况应用上下文中每个方面只有一个单独的实例,Spring支持perthis和pertarget 模型。
基于模式的AOP
如果不能使用JDK5,可以使用基于XML格式的配置,Spring通过使用aop标记提供对方面定义的支持。在Spring配置中,所有的方面和advisor元素都必须放在一个<aop:config>元素内。</aop:config>
在基于模式的AOP中,Spring仅支持单例模型。要使用AOP标记,必须在配置文件中引入相应的 XSD文件如下:
xml 代码
1. xsi:schemaLocation=" http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd"
声明方面
使用模式支持,一个方面只是Spring应用上下文中的一个Bean实例,状态和行为由这个Bean实例捕获,切入点和advice信息用XML描述。示例如下:
xml 代码
1. <aop:config>
2. <aop:aspect
3. ...
4. aop:aspect>
5. aop:config>
6. <bean id="
7. ...
8. bean>
声明Pointcut
Pointcut可以直接声明在方面内部,也可以直接声明在<aop:config>元素内,可以供多个方面使用。</aop:config>
xml 代码
1. <aop:config>
2. <aop:pointcut id="businessService"
3. Aspect Oriented Programming with Spring
4. Spring Framework (2.0.2) 118
5. expression="execution(* com.xyz.myapp.service.*.*(..))"/>
6. aop:config>
声明advice
下面的示例声明一个before类型的advice
xml 代码
1. <aop:aspect id="beforeExample" ref="aBean">
2. <aop:before
3. pointcut-ref="dataAccessOperation"
4. method="doAccessCheck"/>
5. ...
6. aop:aspect>
声明introduction
下面的示例声明一个introduction
xml 代码
1. <aop:aspect id="usageTrackerAspect" ref="usageTracking">
2. <aop:declare-parents
3. types-matching="com.xzy.myapp.service.*+",
4. implement-interface="UsageTracked"
5. default-impl="com.xyz.myapp.service.tracking.DefaultUsageTracked"/>
6. <aop:before
7. pointcut="com.xyz.myapp.SystemArchitecture.businessService()
8. and this(usageTracked)"
9. method="recordUsage"/>
10. aop:aspect>
Advisor
Advisor是Spring特有的概念,其实就是Pointcut和Advice的容器。
@AspectJ/XML
基于Schema的AOP定义比较清晰明了,而且很容易通过配置文件修改,也有两个缺点:
Schema方式中关于方面的信息由分布在XML文件和Bean实例中,而@Aspectj方式,所有关于方面的信息都集中在一个模块的所有Bean实例中。
Schema方式仅支持方面的单例模型,@Aspectj方式支持组合命名的Pointcut。
代理机制
Spring AOP缺省使用JDK的动态代理机制,只能针对接口代理,如果需要针对类代理,可以使用CGLIB代理,要在AOP中强制使用CGLIB代理,可以设置 proxy-target-class属性如下:
xml 代码
1. <aop:config proxy-target-class="true">
2. <!-- other beans defined here... -->
3. aop:config>
使用CGLIB代理的一个问题是final方法不能被代理,因为final方法不能重写。
Spring框架使用代理机制,会导致调用自身方法的方法不能被代理,下面这种情况下,foo方法就不能被代理。
[/code]