1. 什么是FactoryBean

FactoryBean本质上是一种Bean,只是它可以产生其他的Bean,比较特殊。在上下文getBean的时候,如果传入FactoryBean的名称,得到的是FactoryBean生产的产品,而不是FactoryBean。如果要获得FactoryBean自身,那么传入的FactoryBean名称前面要加上&字符。

2. 一个小例子

 package com.khlin.my.test;

 import org.springframework.beans.factory.FactoryBean;
import java.util.Date; public class DefaultFactoryBean implements FactoryBean<Date> {
public Date getObject() throws Exception {
return new Date();
} public Class<?> getObjectType() {
return Date.class;
} public boolean isSingleton() {
return true;
}
}
<bean id="defaultFactoryBean" class="com.khlin.my.test.DefaultFactoryBean"/>

再来个启动类

 package com.khlin.my.test;

 import org.springframework.beans.factory.FactoryBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; import java.util.Date; public class FactoryBeanTest { public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
Date date = (Date) applicationContext.getBean("defaultFactoryBean");
System.out.println(date);
DefaultFactoryBean factoryBean = (DefaultFactoryBean) applicationContext.getBean("&defaultFactoryBean");
}
}

可以看到,如果要获得产品,直接使用FactoryBean的名称。如果要获得工厂,则要在前面加上&

3. 原理分析

在getBean方法里,层层调用,到了AbstractBeanFactory的getObjectForBeanInstance方法,无论是获取产品还是工厂,都是先取出工厂类的Bean。

然后对其进行类型判断。

3.1 如果是FactoryBean并且名称不是以&开头,说明要取出产品。先是从缓存里取出产品,如果产品不存在,从工厂类中取出。

对于单例的产品,会存储在FactoryBeanRegistrySupport的factoryBeanObjectCache里。这是一个Map,由工厂名映射到产品。

第一次获取,肯定是为空。那么就会调用getObjectFromFactoryBean方法。在这个方法里,会先调用工厂的getObject()方法,最后放入缓存中。这样后面取产品,就可以从缓存中取出,实现了单例的语义。

要注意的是,虽然产品是通过getObject()直接得到的,不是Spring Bean,但中间还是用BeanPostProcessor参与了其生命周期。

最终会进入到AbstractAutowireCapableBeanFactory类。这里只调用了After的方法,没有调用Before.

对于多例,getObjectForBeanInstance方法里,从缓存里取不出产品,同样调用getObjectFromFactoryBean。

不同的是这次走入另一条分支。仍是调用getObject生产产品,再用BeanPostProcessors后处理,不过这次没有将其放入缓存。这也符合多例的语义,每次获得的都是新的对象。

3.2 如果是FactoryBean但不是以&开头,只能走到最后一个分支,直接取出FactoryBean对象。

重温上面的代码图,即走入最后一个分支,return beanInstance.

4. 总结

单例的产品会放在缓存里,键为工厂名,值为产品。

多例的产品不会放在缓存。

创建产品是直接调用工厂的getObject方法,因此产品没有Spring Bean的生命周期。

每次创建完一个产品,会调用所有注册的BeanPostProcessors的postProcessAfterInitialization方法,参与到产品初始化的生命周期。

简单分析FactoryBean的更多相关文章

  1. 简单分析JavaScript中的面向对象

    初学JavaScript的时候有人会认为JavaScript不是一门面向对象的语言,因为JS是没有类的概念的,但是这并不代表JavaScript没有对象的存在,而且JavaScript也提供了其它的方 ...

  2. CSipSimple 简单分析

    简介 CSipSimple是一款可以在android手机上使用的支持sip的网络电话软件,可以在上面设置使用callda网络电话.连接使用方式最好是使用wifi,或者3g这样上网速度快,打起电话来效果 ...

  3. C#中异常:“The type initializer to throw an exception(类型初始值设定项引发异常)”的简单分析与解决方法

    对于C#中异常:“The type initializer to throw an exception(类型初始值设定项引发异常)”的简单分析,目前本人分析两种情况,如下: 情况一: 借鉴麒麟.NET ...

  4. 透过byte数组简单分析Java序列化、Kryo、ProtoBuf序列化

    序列化在高性能网络编程.分布式系统开发中是举足轻重的之前有用过Java序列化.ProtocolBuffer等,在这篇文章这里中简单分析序列化后的byte数组观察各种序列化的差异与性能,这里主要分析Ja ...

  5. 简单分析Java的HashMap.entrySet()的实现

    关于Java的HashMap.entrySet(),文档是这样描述的:这个方法返回一个Set,这个Set是HashMap的视图,对Map的操作会在Set上反映出来,反过来也是.原文是 Returns ...

  6. Ffmpeg解析media容器过程/ ffmpeg 源代码简单分析 : av_read_frame()

    ffmpeg 源代码简单分析 : av_read_frame() http://blog.csdn.net/leixiaohua1020/article/details/12678577 ffmpeg ...

  7. FFmpeg的HEVC解码器源码简单分析:解析器(Parser)部分

    ===================================================== HEVC源码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpeg ...

  8. FFmpeg资料来源简单分析:libswscale的sws_getContext()

    ===================================================== FFmpeg库函数的源代码的分析文章: [骨架] FFmpeg源码结构图 - 解码 FFmp ...

  9. wp7之换肤原理简单分析

    wp7之换肤原理简单分析 纠结很久...感觉勉强过得去啦.还望各位大牛指点江山 百度找到这篇参考文章http://www.cnblogs.com/sonyye/archive/2012/03/12/2 ...

随机推荐

  1. expdp / impdp 用法详解 ,和exp / imp 的区别

    一  关于expdp和impdp     使用EXPDP和IMPDP时应该注意的事项:EXP和IMP是客户端工具程序,它们既可以在客户端使用,也可以在服务端使用.EXPDP和IMPDP是服务端的工具程 ...

  2. jdbc相比于hibernate的弊端

    1.编程人员必须既懂Java语言,又懂SQL语言,才能编写数据库访问代码.(感觉用不用hibernate,SQL都要会呀) 2.程序代码中嵌入大量字符串形式的SQL语句,降低了程序代码的可读性. 3. ...

  3. Python generator 类型

    场景: 使用gurobi求解优化问题时,遇到quicksum()函数用法如下: quicksum(mu[i] for i in range(n)) 读着很流畅而且好像并没什么问题欸,但 mu[i] f ...

  4. 阶段5 3.微服务项目【学成在线】_day05 消息中间件RabbitMQ_12.RabbitMQ研究-工作模式-统配符工作模式测试

    路由模式: 1.每个消费者监听自己的队列,并且设置带统配符的routingkey. 2.生产者将消息发给broker,由交换机根据routingkey来转发消息到指定的队列. 创建测试用例 交换机的名 ...

  5. OpenStack Manila发展动态系列--Austin峰会

    1 Manila Mitaka版本概述 在Austin峰会上介绍到,Manila Mitaka发布版本Driver个数达到了18个, M版本新加入14家公司(中国公司继华为之后又有99cloud等公司 ...

  6. anconda + python 3.6安装(以前的anconda,anaconda和python版本对应关系)

    anconda + python 3.6安装 anaconda + python3.6安装安装老版本带python3.6的anaconda安装老版本带python3.6的anacondaanacond ...

  7. DataFrame执行groupby聚合操作后,如何继续保持DataFrame对象而不变成Series对象

    刚接触pandas不久,在处理特征时,碰到一个恶心的问题:用groupby聚合后,之前的dataframe对象变成了series对象,聚合的字段变成了索引index,导致获取这些字段时很麻烦,后面发现 ...

  8. unity三维地球模型生成

    准备一张贴图   创建材质球 球面坐标系转直角坐标系 x=rsinθcosφ. y=rsinθsinφ. z=rcosθ. 效果如下   脚本如下 using System.Collections; ...

  9. systemctl daemon-reload

    systemctl daemon-reload: 重新加载某个服务的配置文件,如果新安装了一个服务,归属于 systemctl 管理,要是新服务的服务程序配置文件生效,需重新加载. init 和 sy ...

  10. sql注入01

    1.SLQ:操作数据 数据库结构 create/drop database create/alter/drop table 数据库内容 insert values语句 insert into tabl ...