6.3. Spring的Advisor API

在Spring中,Advisor是一个切面,它只包含与切入点表达式关联的单个advice对象。
除了introductions的特殊情况外,任何Advisor都可以与任何通知一起使用。org.springframework.aop.support.DefaultPointcutAdvisor是最常用的Advisor类。它可以与MethodInterceptor、BeforeAdvice或ThrowsAdvice一起使用。
在同一个AOP代理中,可以在Spring中混合Advisor和Advice类型。例如,在一个代理配置中,您可以使用围绕通知,throws 通知,before通知。Spring自动创建必要的拦截器链。

6.4. 使用ProxyFactoryBean创建AOP代理

如果将Spring IOC容器(ApplicationContext或BeanFactory)用于业务对象,您希望使用Spring的AOP FactoryBean实现之一。(请记住,工厂bean引入了一个间接层,允许它创建不同类型的对象。)

Spring AOP支持还使用了工厂级的bean。
在Spring中创建AOP代理的基本方法是使用org.springframework.aop.framework.ProxyFactoryBean。这可以完全控制切入点、任何适用的通知以及它们的顺序。但是,如果您不需要这样的控制,可以选择更简单的选项。

6.4.1. 基础

ProxyFactoryBean与其他Spring FactoryBean实现一样,引入了一个间接级别。如果定义一个名为foo的ProxyFactoryBean ,则引用foo的对象不会看到ProxyFactoryBean 实例本身,而是由ProxyFactoryBean 中getObject() 方法的实现创建的对象。此方法创建包装目标对象的AOP代理。

使用ProxyFactoryBean或其他具有IOC意识的类来创建AOP代理最重要的好处之一是,通知和切入点也可以由IOC管理。这是一个强大的特性,使得某些方法很难用其他AOP框架实现。例如,一个通知本身可以引用应用程序对象(除了目标之外,目标应该在任何AOP框架中都可用),这得益于依赖注入提供的所有可插入性。

6.4.2. JavaBean属性

与Spring提供的大多数FactoryBean实现相同,proxyFactoryBean类本身就是一个JavaBean。其属性用于:

  • 指定要代理的目标。
  • 指定是否使用CGLIB。

一些关键属性继承自org.springframework.aop.framework.ProxyConfig (Spring中所有AOP代理工厂的超类)。这些关键属性包括:

  • proxyTargetClass:如果要代理目标类,而不是目标类的接口,则为true。如果将此属性值设置为true,则会创建CGLIB代理(也可参见基于JDK和CGLIB的代理)。
  • optimize:控制是否对通过CGLIB创建的代理应用积极的优化。除非您完全理解相关AOP代理如何处理优化,否则不应随意使用此设置。这目前仅用于CGLIB代理。它对JDK动态代理没有影响。
  • CGLIB:如果代理配置被冻结,则不再允许更改配置。这对于轻微的优化和不希望调用方能够在创建代理后操纵代理的情况都很有用。此属性的默认值为false,因此允许进行更改(例如添加其他通知)。
  • exposeProxy:确定当前代理是否应在ThreadLocal中公开,以便目标可以访问它。如果目标需要获取代理,并且 exposeProxy 属性设置为true,则该目标可以使用 AopContext.currentProxy()方法。

ProxyFactoryBean特有的其他属性包括:
proxyInterfaces:字符串接口名称的数组。如果没有提供,将使用目标类的CGLIB代理(也可参见基于JDK和CGLIB的代理)。
interceptorNames:要应用的顾问、拦截器或其他通知名称的字符串数组。顺序非常重要,以先到先得的方式提供。也就是说,列表中的第一个拦截器是第一个能够拦截调用的拦截器。
名称是当前工厂中的bean名称,包括来自祖先工厂的bean名称。这里不能提到bean引用,因为这样做会导致 ProxyFactoryBean忽略通知里的singleton设置。
您可以附加一个带有星号(*)的拦截器名称。这样做会导致所有Advisor Bean都应用,其名称以要应用的星号前面的部分开头。
singleton:工厂是否应该返回单例对象,无论调用getObject()方法的频率如何。一些FactoryBean实现提供了这样的方法。默认值为true。如果您想使用有状态的通知(例如,对于有状态的混合),可以使用原型通知,他的singleton值为false。

6.4.3. 基于JDK和CGLIB的代理

本节是关于ProxyFactoryBean如何选择为特定目标对象(要代理)创建基于JDK的代理或基于CGLIB的代理的最终文档。
ProxyFactoryBean在创建基于JDK或CGLIB的代理方面的行为在Spring的1.2.x和2.0版本之间发生了变化。在自动检测接口方面,ProxyFactoryBean现在显示出与TransactionProxyFactoryBean类相似的语义。

如果要代理的目标对象的类(以下简称为目标类)不实现任何接口,则创建基于CGLIB的代理。这是最简单的场景,因为JDK代理是基于接口的,没有接口意味着JDK代理甚至不可能实现。您可以插入目标bean并通过设置 interceptorNames属性指定拦截器列表。注意,即使proxyTargetClass的ProxyFactoryBean 属性设置为false,也会创建基于CGLIB的代理。(这样做毫无意义,最好从bean定义中删除,因为它充其量是多余的,最坏的情况是令人困惑。)

如果目标类实现一个(或多个)接口,则创建的代理的类型取决于ProxyFactoryBean的配置。

如果ProxyFactoryBean的proxyTargetClass属性设置为true,则会创建基于CGLIB的代理。即使ProxyFactoryBean的proxyInterfaces属性已设置为一个或多个完全限定的接口名称,proxyTargetClass属性设置为true的事实也会导致基于cglib的代理生效。

如果ProxyFactoryBean的proxyInterfaces属性已设置为一个或多个完全限定的接口名称,则将创建基于JDK的代理。创建的代理实现在proxyInterfaces属性中指定的所有接口。如果目标类恰好实现了比proxyInterfaces属性中指定的接口多得多的接口,那么这一切都还好,但是这些附加接口不是由返回的代理实现的。

如果尚未设置ProxyFactoryBean的proxyInterfaces属性,但目标类确实实现了一个(或多个)接口,则ProxyFactoryBean会自动检测到目标类确实实现了至少一个接口,并创建了基于JDK的代理。实际代理的接口是目标类实现的所有接口。实际上,这与向proxyInterfaces属性提供目标类实现的每个接口的列表相同。然而,它的工作量明显减少,而且不容易出现排版错误。