使用“自动代理”功能

到目前为止,我们已经考虑过使用 ProxyFactoryBean 或类似的工厂 Bean 显式创建 AOP 代理。

Spring 还允许我们使用“自动代理” Bean 定义,它可以自动代理选定的 Bean 定义。这是建立在 Spring 的“Bean 后置处理器”基础设施之上的,它允许在容器加载时修改任何 Bean 定义。

在这个模型中,您需要在 XML bean 定义文件中设置一些特殊的 bean 定义来配置自动代理基础设施。这使您可以声明符合自动代理条件的目标。您不需要使用 ProxyFactoryBean

有两种方法可以做到这一点

  • 使用一个自动代理创建器,它引用当前上下文中的特定 bean。

  • 自动代理创建的一个特殊情况,值得单独考虑:由源代码级元数据属性驱动的自动代理创建。

自动代理 Bean 定义

本节介绍 org.springframework.aop.framework.autoproxy 包提供的自动代理创建器。

BeanNameAutoProxyCreator

BeanNameAutoProxyCreator 类是一个 BeanPostProcessor,它会自动为名称与字面值或通配符匹配的 bean 创建 AOP 代理。以下示例展示了如何创建一个 BeanNameAutoProxyCreator bean

<bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
	<property name="beanNames" value="jdk*,onlyJdk"/>
	<property name="interceptorNames">
		<list>
			<value>myInterceptor</value>
		</list>
	</property>
</bean>

ProxyFactoryBean 一样,它有一个 interceptorNames 属性,而不是拦截器列表,以允许原型增强器正常工作。名为“拦截器”的可以是增强器或任何增强类型。

与自动代理总体情况一样,使用 BeanNameAutoProxyCreator 的主要目的是将相同的配置一致地应用于多个对象,并最大限度地减少配置量。它是将声明式事务应用于多个对象的流行选择。

名称匹配的 bean 定义(例如,前面示例中的 jdkMyBeanonlyJdk)是具有目标类的普通 bean 定义。BeanNameAutoProxyCreator 会自动创建一个 AOP 代理。相同的增强会应用于所有匹配的 bean。请注意,如果使用增强器(而不是前面示例中的拦截器),则切入点可能对不同的 bean 应用不同。

DefaultAdvisorAutoProxyCreator

一个更通用且功能强大的自动代理创建器是 DefaultAdvisorAutoProxyCreator。它会自动将当前上下文中符合条件的增强器应用于目标对象,而无需在自动代理增强器的 bean 定义中包含特定的 bean 名称。它与 BeanNameAutoProxyCreator 一样,具有配置一致性和避免重复的优点。

使用此机制涉及

  • 指定一个 DefaultAdvisorAutoProxyCreator bean 定义。

  • 在相同或相关上下文中指定任意数量的顾问。请注意,这些必须是顾问,而不是拦截器或其他建议。这是必要的,因为必须有一个切入点来评估,以检查每个建议对候选 bean 定义的资格。

DefaultAdvisorAutoProxyCreator 会自动评估每个顾问中包含的切入点,以查看它应该将哪些(如果有)建议应用于每个业务对象(例如,示例中的 businessObject1businessObject2)。

这意味着可以将任意数量的顾问自动应用于每个业务对象。如果任何顾问中的任何切入点都不匹配业务对象中的任何方法,则不会对该对象进行代理。随着为新业务对象添加 bean 定义,如果需要,它们会自动进行代理。

自动代理通常具有以下优点:它使调用者或依赖项无法获取未经建议的对象。在该 ApplicationContext 上调用 getBean("businessObject1") 会返回一个 AOP 代理,而不是目标业务对象。(前面显示的“内部 bean”习语也提供了此优势。)

以下示例创建了一个 DefaultAdvisorAutoProxyCreator bean 和本节中讨论的其他元素

<bean class="org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator"/>

<bean class="org.springframework.transaction.interceptor.TransactionAttributeSourceAdvisor">
	<property name="transactionInterceptor" ref="transactionInterceptor"/>
</bean>

<bean id="customAdvisor" class="com.mycompany.MyAdvisor"/>

<bean id="businessObject1" class="com.mycompany.BusinessObject1">
	<!-- Properties omitted -->
</bean>

<bean id="businessObject2" class="com.mycompany.BusinessObject2"/>

如果您希望将相同的建议一致地应用于许多业务对象,DefaultAdvisorAutoProxyCreator 非常有用。一旦基础设施定义到位,您就可以添加新的业务对象,而无需包含特定的代理配置。您还可以轻松地添加其他方面(例如,跟踪或性能监控方面),而对配置的更改最少。

DefaultAdvisorAutoProxyCreator 支持过滤(通过使用命名约定,以便只评估某些顾问,这允许在同一个工厂中使用多个配置不同的 AdvisorAutoProxyCreators)和排序。顾问可以实现 org.springframework.core.Ordered 接口,以确保如果这是一个问题,则按正确的顺序排序。前面示例中使用的 TransactionAttributeSourceAdvisor 具有可配置的顺序值。默认设置是无序的。