Spring之FactoryBean

首先要分辨BeanFactory 與 FactoryBean的區別, 兩個名字很像,因此容易搞混html

BeanFactory: 以Factory結尾,表示它是一個工廠類,是用於管理Bean的一個工廠java

FactoryBean:以Bean結尾,表示它是一個Bean,不一樣於普通Bean的是:它是實現了FactoryBean<T>接 口的Bean,根據該Bean的Id從BeanFactory中獲取的其實是FactoryBean的getObject()返回的對象,而不是 FactoryBean自己, 若是要獲取FactoryBean對象,能夠在id前面加一個&符號來獲取。spring


Spring中的Bean有兩種。測試

一種是普通的bean ,好比配置this

[html] view plaincopyspa

  1. <bean id="personService" class="com.spring.service.impl.PersonServiceImpl" scope="prototype">  .net

  2.             <property name="name" value="is_zhoufeng" />  prototype

  3.       </bean>    代理

那個使用BeanFactory根據id personService獲取bean的時候,獲得的對象就是PersonServiceImpl類型的。htm


另一種就是實現了org.springframework.beans.factory.FactoryBean<T>接口的 Bean , 那麼在從BeanFactory中根據定義的id獲取bean的時候,獲取的其實是FactoryBean接口中的getObject()方法返回的對 象。

以Spring提供的ProxyFactoryBean爲例子,配置以下:

[html] view plaincopy

  1. <bean id="personServiceByLog" class="org.springframework.aop.framework.ProxyFactoryBean">  

  2.             <property name="proxyInterfaces">  

  3.                 <list>  

  4.                     <value>com.spring.service.PersonService</value>  

  5.                 </list>  

  6.             </property>  

  7.             <property name="interceptorNames">  

  8.                 <list>  

  9.                     <value>logInteceptor</value>  

  10.                     <value>ZFMethodAdvice</value>  

  11.                 </list>  

  12.             </property>  

  13.             <property name="targetName" value="personService" />    

  14.      </bean>  


那麼在代碼中根據personServiceByLog來獲取的Bean其實是PersonService類型的。 

[java] view plaincopy

  1. @Test   

  2.  public void test01() {  

  3.   

  4.      PersonService ps = context.getBean("personService", PersonService.class);  

  5.   

  6.      ps.sayHello();  

  7.   

  8.      String name = ps.getName();  

  9.   

  10.      System.out.println(name);  

  11.  }  


若是要獲取ProxyFactoryBean自己,能夠以下

[java] view plaincopy

  1. @Test   

  2.  public void test04() {  

  3.      ProxyFactoryBean factoryBean = context.getBean("&personServiceByLog", ProxyFactoryBean.class);  

  4.      PersonService ps = (PersonService) factoryBean.getObject();  

  5.      String name = ps.getName();  

  6.      System.out.println(name);  

  7.   

  8.  }  





本身實現一個FactoryBean, 功能:用來代理一個對象,對該對象的全部方法作一個攔截,在方法調用先後都輸出一行log

[java] view plaincopy

  1. package com.spring.factorybean;  

  2.   

  3. import java.lang.reflect.InvocationHandler;  

  4. import java.lang.reflect.Method;  

  5. import java.lang.reflect.Proxy;  

  6.   

  7. import org.springframework.beans.factory.DisposableBean;  

  8. import org.springframework.beans.factory.FactoryBean;  

  9. import org.springframework.beans.factory.InitializingBean;  

  10.   

  11. public class ZFFactoryBean implements FactoryBean<Object>, InitializingBean, DisposableBean {  

  12.   

  13.     // 被代理對象實現的接口名(在使用Proxy時須要用到,用於決定生成的代理對象類型)  

  14.     private String interfaceName;  

  15.   

  16.     // 被代理的對象  

  17.     private Object target;  

  18.   

  19.     // 生成的代理對象  

  20.     private Object proxyObj;  

  21.   

  22.     public void destroy() throws Exception {  

  23.         System.out.println("distory...");  

  24.     }  

  25.   

  26.     public void afterPropertiesSet() throws Exception {  

  27.   

  28.         proxyObj = Proxy.newProxyInstance(this.getClass().getClassLoader(),  

  29.                                           new Class[] { Class.forName(interfaceName) }, new InvocationHandler() {  

  30.   

  31.                                               public Object invoke(Object proxy, Method method, Object[] args)  

  32.                                                                                                               throws Throwable {  

  33.                                                   System.out.println("method:" + method.getName());  

  34.                                                   System.out.println("Method before...");  

  35.                                                   Object result = method.invoke(target, args);  

  36.                                                   System.out.println("Method after...");  

  37.                                                   return result;  

  38.                                               }  

  39.                                           });  

  40.   

  41.         System.out.println("afterPropertiesSet");  

  42.     }  

  43.   

  44.     public Object getObject() throws Exception {  

  45.         System.out.println("getObject");  

  46.         return proxyObj;  

  47.     }  

  48.   

  49.     public Class<?> getObjectType() {  

  50.         return proxyObj == null ? Object.class : proxyObj.getClass();  

  51.     }  

  52.   

  53.     public boolean isSingleton() {  

  54.         return true;  

  55.     }  

  56.   

  57.     public String getInterfaceName() {  

  58.         return interfaceName;  

  59.     }  

  60.   

  61.     public void setInterfaceName(String interfaceName) {  

  62.         this.interfaceName = interfaceName;  

  63.     }  

  64.   

  65.     public Object getTarget() {  

  66.         return target;  

  67.     }  

  68.   

  69.     public void setTarget(Object target) {  

  70.         this.target = target;  

  71.     }  

  72.   

  73. }  


而後來試試:

首先這樣定義bean

[java] view plaincopy

  1. <bean id="personService" class="com.spring.service.impl.PersonServiceImpl" scope="prototype">  

  2.             <property name="name" value="is_zhoufeng" />  

  3.       </bean>    

  4.         

  5.       <bean id="zfPersonService" class="com.spring.factorybean.ZFFactoryBean">  

  6.         <property name="interfaceName" value="com.spring.service.PersonService" />  

  7.         <property name="target"  ref="personService"/>  

  8.       </bean>  

而後獲取Bean,並測試。

[java] view plaincopy

  1. @Test   

  2.  public void test06() {  

  3.      PersonService ps = context.getBean("zfPersonService", PersonService.class);  

  4.   

  5.      ps.sayHello();  

  6.   

  7.      String name = ps.getName();  

  8.   

  9.      System.out.println(name);  

  10.  }  


會發現sayHello與getName方法調用先後都有log打印。



上面的ZFBeanFactory只是模仿了ProxyFactoryBean的功能作了一個實現而已。

其實經過FactoryBean這種特色,能夠實現不少有用的功能 。。。

相關文章
相關標籤/搜索