Spring之FactoryBean

简介:

首先要分辨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 plain  copy
 
  在CODE上查看代码片 派生到我的代码片
  1. <bean id="personService" class="com.spring.service.impl.PersonServiceImpl" scope="prototype">  
  2.             <property name="name" value="is_zhoufeng" />  
  3.       </bean>    

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

 

 

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

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

 

[html]  view plain  copy
 
  在CODE上查看代码片 派生到我的代码片
  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 plain  copy
 
  在CODE上查看代码片 派生到我的代码片
  1. @Test  
  2.  public void test01() {  
  3.   
  4.      PersonService ps = context.getBean("personServiceByLog", PersonService.class);  
  5.   
  6.      ps.sayHello();  
  7.   
  8.      String name = ps.getName();  
  9.   
  10.      System.out.println(name);  
  11.  }  


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

 

 

[java]  view plain  copy
 
  在CODE上查看代码片 派生到我的代码片
  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 plain  copy
 
  在CODE上查看代码片 派生到我的代码片
  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 plain  copy
 
  在CODE上查看代码片 派生到我的代码片
  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 plain  copy
 
  在CODE上查看代码片 派生到我的代码片
  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这种特点,可以实现很多有用的功能 。。。


原文链接:[http://wely.iteye.com/blog/2341193]

相关文章
|
7月前
|
Java 关系型数据库 MySQL
Spring5深入浅出篇:Spring中的FactoryBean对象
Spring5深入浅出篇:Spring中的FactoryBean对象
|
Java Spring 容器
Spring中BeanFactory和FactoryBean的区别?
一位工作了4年的小伙伴,去京东面试被问到这样一个问题,Spring中的BeanFactory和FactoryBean有什么区别?因为没有看过源码,当时就感觉这是一个文字游戏,感觉没什么区别? 那今天,我就给大家来聊清楚。另外,往期面试题解析中配套的文档我已经准备好,想获得的可以在我的煮叶简介中找到。好了,我们先来看BeanFactory。
67 0
|
7月前
|
XML Java 数据格式
spring中怎么通过静态工厂和动态工厂获取对象以及怎么通过 FactoryBean 获取对象
spring中怎么通过静态工厂和动态工厂获取对象以及怎么通过 FactoryBean 获取对象
91 0
|
2月前
|
缓存 JavaScript Java
Spring之FactoryBean的处理底层源码分析
本文介绍了Spring框架中FactoryBean的重要作用及其使用方法。通过一个简单的示例展示了如何通过FactoryBean返回一个User对象,并解释了在调用`getBean()`方法时,传入名称前添加`&`符号会改变返回对象类型的原因。进一步深入源码分析,详细说明了`getBean()`方法内部对FactoryBean的处理逻辑,解释了为何添加`&`符号会导致不同的行为。最后,通过具体代码片段展示了这一过程的关键步骤。
Spring之FactoryBean的处理底层源码分析
|
2月前
|
XML 缓存 Java
Spring FactoryBean 的常见使用场景总结
FactoryBean 是 Spring 框架中的一个重要接口,用于自定义 Bean 的创建逻辑。常见使用场景包括: 1. **复杂 Bean 的创建**:如数据源配置。 2. **延迟实例化**:按需创建资源密集型对象。 3. **动态代理**:为 Bean 创建 AOP 代理。 4. **自定义配置**:根据特定配置创建 Bean。 5. **第三方库集成**:利用 FactoryBean 封装外部库的创建过程。
|
7月前
|
Java 数据库连接 API
【Spring】1、Spring 框架的基本使用【读取配置文件、IoC、依赖注入的几种方式、FactoryBean】
【Spring】1、Spring 框架的基本使用【读取配置文件、IoC、依赖注入的几种方式、FactoryBean】
109 0
|
6月前
|
Java Spring
聊聊Spring中两种创建Bean的方式:BeanDefinition.setInstanceSupplier() 和 FactoryBean
聊聊Spring中两种创建Bean的方式:BeanDefinition.setInstanceSupplier() 和 FactoryBean
|
7月前
|
XML Java 数据格式
Spring5源码(8)-BeanFactory和FactoryBean的区别
Spring5源码(8)-BeanFactory和FactoryBean的区别
68 0
|
XML Java 数据格式
Spring中BeanFactory和FactoryBean详解
Spring中BeanFactory和FactoryBean详解
336 1
|
7月前
|
XML Java 数据格式
③【Spring】整合第三框架的常用机制:FactoryBean
③【Spring】整合第三框架的常用机制:FactoryBean
95 0