欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Spring Aop(十三)——ProxyFactoryBean创建代理对象

程序员文章站 2022-05-09 17:05:44
...

ProxyFactoryBean创建代理对象

ProxyFactoryBean实现了Spring的FactoryBean接口,所以它跟Spring中的其它FactoryBean一样,都是基于工厂模式来获取一个bean的。ProxyFactoryBean就是用来获取一个对象的代理对象的FactoryBean。它也是继承自ProxyCreatorSupport类的,所以它的功能基本跟ProxyFactory差不多,只是ProxyFactory是用于编程式的创建代理对象。而ProxyFactoryBean用于在Spring的bean容器中创建基于bean的代理对象。通常一个简单的ProxyFactoryBean配置大概会是如下这样。

<bean id="proxyFactoryBeanTestService" 
  class="org.springframework.aop.framework.ProxyFactoryBean">
  <property name="target"><!-- 指定被代理的对象 -->
    <bean class="com.elim.learn.spring.aop.service.ProxyFactoryBeanTestService"/>
  </property>
  <property name="proxyTargetClass" value="true"/><!-- 指定启用基于Class的代理 -->
  <!-- 指定生成的代理对象需要绑定的Advice或Advisor在bean容器中的名称 -->
  <property name="interceptorNames">
    <list>
      <value>logAroundAdvice</value>
    </list>
  </property>
</bean>

在上面的示例中我们被代理的对象对应的Class是com.elim.learn.spring.aop.service.ProxyFactoryBeanTestService,其是一个没有实现任何接口的Class,所以我们生成的代理对象最终会是基于CGLIB的代理。我们需要指定proxyTargetClass="true",以表示我们是倾向于使用CGLIB代理的,对于上面的配置实际上就是告诉Spring我们要使用CGLIB代理。虽然这里我们不指定proxyTargetClass="true"时,Spring可能也会给我们使用CGLIB代理,为什么这里说是可能呢?因为ProxyFactoryBean默认生成的bean都是单例、且在生成bean时会自动检测被代理对象实现的接口,而且proxyTargetClass默认是false,这种情况下ProxyFactoryBean就会自动检测被代理对象的实现的接口。按理来说我们的bean是一个没有实现接口的bean,Spring给我们去找它实现的接口是找不出来的,但是我们知道Spring的Aop是会自动为我们的对象实现一些接口的。简单的说如果我们的bean容器中配置了其它的Advisor,那么我们指定的target对象有可能就不是一个原始的bean,而是一个已经被Aop代理过的bean对象,这种bean对象,Spring Aop默认会为其实现一个Advised接口。所以使用ProxyFactoryBean时,如果我们的代理对象类是没有实现接口的,或者我们期望生成代理对象时是基于Class的,而不是基于Interface的,我们最好明确的指定proxyTargetClass="true",而不是寄希望于Spring的自动决定机制。 指定被代理对象时,除了可以直接指定target外,我们还可以通过targetName指定被代理对象在bean容器中的bean名称。 在上面的示例中我们通过interceptorNames属性指定了生成的代理对象需要应用的Advisor/Advice对应于bean容器中的bean的名称。跟ProxyFactory一样,如果我们指定的是Advice对象,则其会转换为一个匹配所有方法调用的Advisor与代理对象绑定。 在指定intercepterNames时我们也可以通过“*”指定所有beanName以XX开始的Advisor/Advice,如我们的bean容器中同时拥有“abc1Advisor”、“abc2Advisor”两个Advisor,我们期望创建的ProxyFactoryBean同时应用这两个Advisor,那我们可以不用单独指定两次,而是一次性把interceptorNames指定的一个beanName为“abc*”。需要注意的是“*”只能定义在beanName的末端。

<bean id="proxyFactoryBeanTestService" 
  class="org.springframework.aop.framework.ProxyFactoryBean">
  <property name="target"><!-- 指定被代理的对象 -->
    <bean class=" com.elim.learn.spring.aop.service.ProxyFactoryBeanTestService"/>
  </property>
  <property name="proxyTargetClass" value="true"/><!-- 指定启用基于Class的代理 -->
  <!-- 指定生成的代理对象需要绑定的Advice或Advisor在bean容器中的名称 -->
  <property name="interceptorNames">
    <list>
      <value>abc*</value>
    </list>
  </property>
</bean>

其它配置信息

  • exposeProxy:属于从ProxyCreatorSupport继承过来的属性,用于定义是否需要在调用代理对象时把代理对象发布到AopContext,默认是false
  • singleton:用来指定ProxyFactoryBean生成的bean是否是单例的,默认是true。该值对应于FactoryBean的isSingleton()接口方法的返回值。
  • frozen:属于从ProxyCreatorSupport继承过来的属性,用于指定代理对象被创建后是否还允许更改代理配置,通过Advised接口更改。true表示不允许,默认是false。
  • autodetectInterfaces:表示是否在生成代理对象时需要启用自动检测被代理对象实现的接口,默认是true
  • proxyInterfaces:基于接口的代理时指定需要代理的接口。
  • interfaces:基于接口的代理时指定需要代理的接口,属于从ProxyCreatorSupport继承过来的。 关于ProxyFactoryBean的更多配置项信息可以参考Spring的API文档或ProxyFactoryBean的源代码。
    (注:本文是基于Spring4.1.0所写,Elim写于2017年5月10日)