`

Spring之FactoryBean 用法

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

2.实例
以下SimpleFactoryBean类实现了FactoryBean接口中的三个方法。 并将该类配置在XML中。
Java代码  收藏代码
public class SimpleFactoryBean implements FactoryBean { 
        private boolean flag; 
     
        public Object getObject() throws Exception { 
            if (flag) { 
                return new Date(); 
            } 
            return new String("false"); 
        } 
     
        @SuppressWarnings("unchecked") 
        public Class getObjectType() { 
            return flag ? Date.class : String.class; 
        } 
     
        public boolean isSingleton() { 
            return false; 
        } 
     
        public void setFlag(boolean flag) { 
            this.flag = flag; 
        } 
    } 
     
    <bean id="factoryBeanOne" class="com.study.demo.factorybean.SimpleFactoryBean" > 
        <property name="flag"> 
            <value>true</value> 
        </property> 
    </bean> 
     
    <bean id="factoryBeanTwo" class="com.study.demo.factorybean.SimpleFactoryBean" > 
        <property name="flag"> 
            <value>false</value> 
        </property> 
    </bean> 
     
    public class MainTest { 
        public static void main(String[] args) { 
            Resource res = new ClassPathResource("bean.xml"); 
            BeanFactory factory = new XmlBeanFactory(res); 
            System.out.println(factory.getBean("factoryBeanOne").getClass()); 
            System.out.println(factory.getBean("factoryBeanTwo").getClass()); 
        } 
    } 


通过简单的测试可知,该类输出如下:
class java.util.Date
class java.lang.String
也就是说,容器通过getBean方法返回的不是FactoryBean本身,而是FactoryBean实现类中getObject()方法所返回的对象。

3.FactoryBean的扩展应用
以Spring集成 ehcache 为例,看下org.springframework.cache.ehcache.EhCacheManagerFactoryBean类与org.springframework.cache.ehcache.EhCacheFactoryBean类。EhCacheManagerFactoryBean类中的getObject()和getObjectType() 方法返回的如下:
Java代码  收藏代码
public Object getObject() { 
            return this.cacheManager; 
        } 
     
    public Class getObjectType() { 
            return (this.cacheManager != null ? this.cacheManager.getClass() : CacheManager.class); 
        } 


EhCacheFactoryBean类中的getObject()和getObjectType() 方法返回的如下:
Java代码  收藏代码
public Object getObject() { 
            return this.cache; 
        } 
     
        public Class getObjectType() { 
            return (this.cache != null ? this.cache.getClass() : Ehcache.class); 
        } 

有了这两个FactoryBean,在Spring容器中使用ehcache就变得很简单了。配置如下:
Java代码  收藏代码
<bean id="cacheManager" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean">   
              <property name="configLocation">   
              <value>classpath:ehcache.xml</value>   
              </property>   
        </bean>   
     
        <bean id="levelOneCache" class="org.springframework.cache.ehcache.EhCacheFactoryBean">   
              <property name="cacheManager">   
                    <ref local="cacheManager" />   
              </property>   
              <property name="cacheName">   
                    <value>levelOneCache</value>   
              </property>   
        </bean> 

应用代码如下:
Java代码  收藏代码
public class MainTest { 
            public static void main(String[] args) { 
                Resource res = new ClassPathResource("bean.xml"); 
                BeanFactory factory = new XmlBeanFactory(res); 
                //取到CacheManager类的实例  
                CacheManager cacheManager = (CacheManager) factory 
                        .getBean("cacheManager"); 
                //取到Cache类的实例  
                Cache levelOneCache = cacheManager.getCache("levelOneCache"); 
            } 
        } 
         


=================================================================================




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

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

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


Spring中的Bean有两种。

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

[html] view plaincopy
<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为例子,配置如下:

[html] view plaincopy
<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类型的。
[java] view plaincopy
@Test 
public void test01() { 
 
     PersonService ps = context.getBean("personService", PersonService.class); 
 
     ps.sayHello(); 
 
     String name = ps.getName(); 
 
     System.out.println(name); 


如果要获取ProxyFactoryBean本身,可以如下
[java] view plaincopy
@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

[java] view plaincopy
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

[java] view plaincopy
<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,并测试。
[java] view plaincopy
@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这种特点,可以实现很多有用的功能 。。。

分享到:
评论

相关推荐

    spring的FactoryBean增强我们的目标对象.rar

    在标题“spring的FactoryBean增强我们的目标对象.rar”中,提到的核心概念就是如何利用`FactoryBean`来增强目标对象,从而在方法调用时使用代理类而不是原始对象。 `FactoryBean`是Spring中用于对象创建的一种高级...

    spring源码解析之FactoryBean相关测试代码demo

    Spring容器会调用`FactoryBean`的`getObject()`方法来获取实际的对象,而不是直接实例化`FactoryBean`本身。 下面我们将逐步分析`FactoryBean`的相关测试代码: 1. **配置文件(spring.xml)** 在Spring的XML配置...

    spring中FactoryBean中的getObject()方法实例解析

    Spring 中 FactoryBean 中的 getObject() 方法实例解析 Spring 框架中 FactoryBean 是一个非常重要的概念,它提供了一种创建和管理 Bean 的机制。在 Spring 中,FactoryBean 是一个特殊的 Bean,它可以创建其他 ...

    Spring中的FactoryBean.rar

    综上所述,"Spring中的FactoryBean.rar"文件所涵盖的内容可能涉及FactoryBean的接口定义、应用场景、使用方法以及如何自定义FactoryBean等方面。通过学习这部分内容,开发者能够更好地理解Spring IoC容器的工作原理...

    详解Spring中的FactoryBean

    在Spring配置中,当我们引用一个`FactoryBean`时,Spring会识别到这个特殊的Bean,并调用`FactoryBean`的方法来获取最终的实例。例如,有一个`Person`类,它需要一个`Car`对象。在XML配置中,我们不需要直接声明`Car...

    简单了解Spring中BeanFactory与FactoryBean的区别

    例如,在Spring MVC中,我们可以使用BeanFactory来管理Controller Bean的生命周期,而使用FactoryBean来生成视图解析器的实例。 BeanFactory和FactoryBean都是Spring框架中的重要组件,它们提供了不同的功能和机制...

    spring中的FactoryBean代码示例

    Spring中的FactoryBean代码示例 在Spring框架中,FactoryBean是一种特殊的Bean,它可以生成其他Bean的实例。今天我们来了解一下FactoryBean的实现和使用。 首先,让我们从SessionFactory说起。在使用SSH集成开发时...

    FactoryBean代码最简实现

    本篇我们将深入探讨`FactoryBean`的最简实现及其在Spring配置文件中的使用,帮助你理解其实例化过程。 `FactoryBean`是Spring框架提供的一种扩展点,它允许我们自定义对象的创建逻辑,而不仅仅是简单的通过`new`...

    深入了解Spring中的FactoryBean

    深入了解Spring中的FactoryBean FactoryBean是一个非常重要的概念,在Spring框架中扮演着核心角色。...了解FactoryBean的概念和应用,可以帮助我们更好地使用Spring框架,开发更加复杂和灵活的业务系统。

    spring如何通过FactoryBean配置Bean

    通过这个示例,我们可以看到 FactoryBean 的强大之处。它可以将复杂的实例化逻辑封装起来,使得 Bean 的配置更加灵活和灵活。同时,它还可以帮助我们实现单例模式,使得我们的应用程序更加高效和稳定。

    Spring BeanFactory和FactoryBean区别解析

    相比之下,FactoryBean是一个特殊的bean,它可以返回bean的实例的工厂bean,通过实现该接口可以对bean进行一些额外的操作,例如根据不同的配置类型返回不同类型的bean,简化xml配置等。在使用上也有些特殊,...

    Spring中BeanFactory与FactoryBean接口的区别详解

    在这个例子中,`UserServiceFactoryBean`实现了FactoryBean,当从Spring容器中请求`IUserService`类型的Bean时,实际上会通过`UserServiceFactoryBean`的`getObject()`方法返回一个`UserServiceImpl`实例。...

    SSH笔记-通过实现FactoryBean接口来创建 Bean

    当你在Spring配置文件中声明一个Bean,并指定它的类型为`FactoryBean`的实现类时,Spring不会直接调用无参构造函数来创建Bean,而是调用`FactoryBean`的方法来获取Bean实例。 首先,我们需要了解`FactoryBean`的...

    spring6pdf详细讲解

    Spring 提供了多种方式来实例化和管理 Bean,包括通过构造方法、简单工厂模式、FactoryBean 等。 Spring 还提供了多种范围的 Bean,包括 singleton、prototype 等。 Spring 的 IoC 容器是框架的核心组件,它负责...

    mybatis-spring-1.3.3.jar官方下载

    MyBatis-Spring 是一个将 MyBatis ORM 框架与 Spring 框架集成的库,使得在 Spring 应用中使用 MyBatis 变得更加方便。mybatis-spring-1.3.3.jar 文件是这个集成库的一个版本,提供了对 MyBatis 1.3.3 和 Spring 的...

    SpringQuartz的使用文档

    Spring对Quartz进行了集成,使其更易于在Spring应用程序中使用。Spring提供了`FactoryBean`类,如`SchedulerFactoryBean`,用于创建和管理Quartz的`Scheduler`实例。这使得我们可以利用Spring的依赖注入(DI)和管理...

    68-BeanFactory与FactoryBean1

    如果我们想获取 BeanFactory 的实例,应该使用 getBean("&beanA"),其中"&"符号表示获取 FactoryBean 的实例,而不是 getObject() 方法返回的对象。 在实际工作中,FactoryBean 是非常有用的,可以用来实例化复杂...

    FactoryBean.zip

    `FactoryBean`的使用使得我们可以控制对象的创建过程,甚至可以在对象被Spring管理的同时,添加额外的初始化或者装配步骤。在这个"FactoryBean.zip"中,很可能是包含了关于如何使用`FactoryBean`来实现特定功能的...

    mybatis-spring-1.0.1-bundle.zip

    它扩展了 Spring 的 FactoryBean 接口,可以直接在 Spring 配置文件中声明,通过配置数据源、MyBatis 的配置文件等参数,轻松创建 SqlSessionFactory。 2. SqlSessionTemplate:这是一个线程安全的 SqlSession 实现...

    mybatis,mybatis-spring

    - MapperFactoryBean:实现了Spring的FactoryBean接口,为每个Mapper接口创建一个代理对象,调用Mapper接口的方法时,会自动执行对应的SQL。 4. **MyBatis-Spring的使用** - 配置:在Spring配置文件中添加...

Global site tag (gtag.js) - Google Analytics