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这种特点,可以实现很多有用的功能 。。。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值