Spring的FactoryBean接口理解

1.Spring中有两种类型的Bean,一种是普通Bean,另一种是工厂Bean,即FactoryBean,这两种Bean都被容器管理,但工厂Bean跟普通Bean不同,其返回的对象不是指定类的一个实例,其返回的是该FactoryBean的getObject方法所返回的对象。在Spring框架内部,有很多地方有FactoryBean的实现类,它们在很多应用如(Spring的AOP、ORM、事务管理)及与其它第三框架(hibernate丶mybatis丶jpa...)集成时都有体现

 FactoryBean的继承实现体系:

3..要分辨BeanFactory 与 FactoryBean的区别, 两个名字很像,所以容易搞混

BeanFactory: 以Factory结尾,表示它是一个工厂类,是用于管理Bean的一个工厂

FactoryBean:以Bean结尾,表示它是一个Bean,不同于普通Bean的是:它是实现了FactoryBean<T>接口的Bean,根据该Bean的Id从BeanFactory中获取的实际上是FactoryBean的getObject()返回的对象,而不是FactoryBean本身, 如果要获取FactoryBean对象,可以在id前面加一个&符号来获取。

Spring中的Bean有两种。

一种是普通的bean ,比如配置

<bean id="personService" class="com.spring.service.impl.PersonServiceImpl" scope="prototype">  
            <property name="name" value="is_zhoufeng" />  
      </bean>    

那个使用BeanFactory根据id personService获取bean的时候,得到的对象就是PersonServiceImpl类型的。

另外一种就是实现了org.springframework.beans.factory.FactoryBean<T>接口的Bean , 那么在从BeanFactory中根据定义的id获取bean的时候,获取的实际上是FactoryBean接口中的getObject()方法返回的对象。

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

<bean id="personServiceByLog" class="org.springframework.aop.framework.ProxyFactoryBean">  
            <property name="proxyInterfaces">  
                <list>  
                    <value>com.spring.service.PersonService</value>  
                </list>  
            </property>  
            <property name="interceptorNames">  
                <list>  
                    <value>logInteceptor</value>  
                    <value>ZFMethodAdvice</value>  
                </list>  
            </property>  
            <property name="targetName" value="personService" />    
     </bean>  


那么在代码中根据personServiceByLog来获取的Bean实际上是PersonService类型的。 

@Test  
 public void test01() {  
  
     PersonService ps = context.getBean("personServiceByLog", PersonService.class);  
  
     ps.sayHello();  
  
     String name = ps.getName();  
  
     System.out.println(name);  
 }  


如果要获取ProxyFactoryBean本身,可以如下

@Test  
 public void test04() {  
     ProxyFactoryBean factoryBean = context.getBean("&personServiceByLog", ProxyFactoryBean.class);  
     PersonService ps = (PersonService) factoryBean.getObject();  
     String name = ps.getName();  
     System.out.println(name);  
  
 }  

自己实现一个FactoryBean, 功能:用来代理一个对象,对该对象的所有方法做一个拦截,在方法调用前后都输出一行log

package com.spring.factorybean;  
  
import java.lang.reflect.InvocationHandler;  
import java.lang.reflect.Method;  
import java.lang.reflect.Proxy;  
  
import org.springframework.beans.factory.DisposableBean;  
import org.springframework.beans.factory.FactoryBean;  
import org.springframework.beans.factory.InitializingBean;  
  
public class ZFFactoryBean implements FactoryBean<Object>, InitializingBean, DisposableBean {  
  
    // 被代理对象实现的接口名(在使用Proxy时需要用到,用于决定生成的代理对象类型)  
    private String interfaceName;  
  
    // 被代理的对象  
    private Object target;  
  
    // 生成的代理对象  
    private Object proxyObj;  
  
    public void destroy() throws Exception {  
        System.out.println("distory...");  
    }  
  
    public void afterPropertiesSet() throws Exception {  
  
        proxyObj = Proxy.newProxyInstance(this.getClass().getClassLoader(),  
                                          new Class[] { Class.forName(interfaceName) }, new InvocationHandler() {  
  
                   public Object invoke(Object proxy, Method method, Object[] args)  
                                                                                                       throws Throwable {  
                         System.out.println("method:" + method.getName());  
                                          
                         System.out.println("Method before...");  
                         Object result = method.invoke(target, args);  
                          System.out.println("Method after...");  
                         return result;  
                                              }  
                                          });  
  
        System.out.println("afterPropertiesSet");  
    }  
  
    public Object getObject() throws Exception {  
        System.out.println("getObject");  
        return proxyObj;  
    }  
  
    public Class<?> getObjectType() {  
        return proxyObj == null ? Object.class : proxyObj.getClass();  
    }  
  
    public boolean isSingleton() {  
        return true;  
    }  
  
    public String getInterfaceName() {  
        return interfaceName;  
    }  
  
    public void setInterfaceName(String interfaceName) {  
        this.interfaceName = interfaceName;  
    }  
  
    public Object getTarget() {  
        return target;  
    }  
  
    public void setTarget(Object target) {  
        this.target = target;  
    }  
  
}  

然后来试试:

首先这样定义bean

<bean id="personService" class="com.spring.service.impl.PersonServiceImpl" scope="prototype">  
            <property name="name" value="is_zhoufeng" />  
      </bean>    
        
      <bean id="zfPersonService" class="com.spring.factorybean.ZFFactoryBean">  
        <property name="interfaceName" value="com.spring.service.PersonService" />  
        <property name="target"  ref="personService"/>  
      </bean>  

然后获取Bean,并测试。

@Test  
 public void test06() {  
     PersonService ps = context.getBean("zfPersonService", PersonService.class);  
  
     ps.sayHello();  
  
     String name = ps.getName();  
  
     System.out.println(name);  
 }  

会发现sayHello与getName方法调用前后都有log打印。

上面的ZFBeanFactory只是模仿了ProxyFactoryBean的功能做了一个实现而已。

其实通过FactoryBean这种特点,可以实现很多有用的功能 。。。

 

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值