1、是什么

FactoryBean的源码比较简单,大家可以细读下其注释。

/*** 实现此接口的bean不能用作普通bean。此bean暴露的对象是通过getObject()创建的对象,而不是它自身*/
public interface FactoryBean<T> {/*** 返回此工厂管理的对象的实例(可能是共享的或独立的,取决于isSingleton()的返回值)*/@NullableT getObject() throws Exception;/*** 返回此FactoryBean创建的对象类型,*/@NullableClass<?> getObjectType();/*** 该工厂管理的对象是否为单例?* 如果是(return true),getObject()总是返回同一个共享的实例,该对象会被BeanFactory缓存起来* 如果是(return false),getObject()返回独立的实例* 一般情况下返回true*/default boolean isSingleton() {return true;}}

  说的简单点,FactoryBean是BeanFactory支持的、用来暴露bean实例的接口

2、有什么用

平常比较常用的配置bean方式有哪几种?

1、基于XML的配置方式

    在xml文件中配置,例如

<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.1.xsd"> <bean id="user" class="com.lee.factorybean.User"> <property name="name" value="zhangsan"/> </bean> </beans>

spring的发布的第一版就支持,这个大家都知道

  2、基于注解的配置方式

    spring2.5开始支持,例如:@Compoment、@Repository、@Controller、@Service等,平时我们用的挺多的

  3、基于Java类的配置方式

    spring3.0开始支持,也是目前spring推荐的方式,@Configuration结合@Bean,springboot中用的非常多

  一般情况下,Spring通过反射机制利用<bean>的class属性指定实现类实例化Bean,在某些情况下,实例化Bean过程比较复杂,如果按照传统的xml方式,则需要在<bean>中提供大量的配置信息。xml配置方式的灵活性是受限的,这时采用编码的方式可能会得到一个简单的方案。那么编码方式又有哪些了?spring3.0之后,编码的方式有基于注解、基于Java类以及基于FactoryBean,那么在spring2.5之前了,如何用xml方式配置实例化过程比较复杂的Bean?可以采用xml结合FactoryBean来实现,xml中配置FactoryBean,FactoryBean创建我们需要的、实例化过程比较复杂的Bean,示例核心代码如下,从spring容器获取name为user的bean实例,获取到的是User类型的Bean

xml配置

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.1.xsd"><bean id="user" class="com.lee.factorybean.config.UserFactoryBean" /> </beans>

UserFactoryBean

public class UserFactoryBean implements FactoryBean<User> {@Overridepublic User getObject() throws Exception {// 假设User的实例化过程比较复杂,在此处进行User的实例化return new User();}@Overridepublic Class<?> getObjectType() {return User.class;}@Overridepublic boolean isSingleton() {return true;}
}

spring2.5之前,只能通过xml的配置方式将Bean注册到spring管理,但是xml的配置方式又不够灵活,配置实例化过程比较复杂的Bean比较麻烦,所有结合FactoryBean,既能采用编码的方式构建实例化过程比较复杂的Bean,也能将Bean交由Spring管理;spring2.5之后,特别是spring3.0之后,注册实例化过程比较复杂的Bean到spring容器的方式就比较多了(可采用的编码方式比较多),FactoryBean的方式也一直被spring支持。

  说的再简单点,通过FactoryBean可以创建实例化过程比较复杂的Bean,至于我们以何种方式将FactoryBean的实例注册到Spring容器,在不同的spring版本,可以采用不同的方式

怎么用

UserFactoryBean

@Component("user")          // beanName = user
public class UserFactoryBean implements FactoryBean<User> {@Overridepublic User getObject() throws Exception {// 假设User的实例化过程比较复杂,在此处进行User的实例化return new User();}@Overridepublic Class<?> getObjectType() {return User.class;}@Overridepublic boolean isSingleton() {return true;}
}

User

public class User { private Integer id; private String name; private Integer age; }

FactoryBeanTest

@SpringBootTest
public class FactoryBeanTest {@Autowiredprivate BeanFactory beanFactory;@Testpublic void test() {Object user = beanFactory.getBean("user");System.out.println(user.getClass().getName());Object bean = beanFactory.getBean("&user");System.out.println(bean.getClass().getName());}}//输出
com.king.kingoeop.domain.User
com.king.kingoeop.factorybean.UserFactoryBean

我们运行测试用例,发现输出的结果是:User,而不是:UserFactoryBean

为什么

具体问题应该是这样的:上述示例中,为什么从spring容器获取的name为user的实例,其类型是User,而不是UserFactoryBean;抽象的问题:根据FactoryBean实例的name获取的为什么不是FactoryBean实例,而是FactoryBean实例的getObject()返回的对象?

我们就以beanFactory.getBean("user");为断点入口进行源码分析,发现一开始从spring容器获取名为user的bean,类型确实是:UserFactoryBean,但是后面又经过getObjectForBeanInstance来真正获取我们需要的对象。

bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);/*** 获取实例对象* 可能是beanInstance自身,也可能是beanInstance创建的对象(如果beanInstance是FactoryBean类型)*/
protected Object getObjectForBeanInstance(Object beanInstance, String name, String beanName, @Nullable RootBeanDefinition mbd) {// name以工厂引用(&)开头,可以跟下isFactoryDereference方法if (BeanFactoryUtils.isFactoryDereference(name)) {if (beanInstance instanceof NullBean) {return beanInstance;}// 如果name以&开头,而beanInstance不是FactoryBean类型,则抛异常(我们没按spring规则来使用)if (!(beanInstance instanceof FactoryBean)) {throw new BeanIsNotAFactoryException(transformedBeanName(name), beanInstance.getClass());}}// 如果beanInstance不是FactoryBean类型,则直接返回beanInstance// 或者name以&开头,也直接返回beanInstance,说明我们就想获取FactoryBean实例if (!(beanInstance instanceof FactoryBean) || BeanFactoryUtils.isFactoryDereference(name)) {return beanInstance;}Object object = null;if (mbd == null) {object = getCachedObjectForFactoryBean(beanName);}if (object == null) {// 此时beanInstance是FactoryBean类型,而name又不是以&开头; 这是我们示例工程的情况,也是最普通、用的最多的情况// 将beanInstance强转成FactoryBean类型FactoryBean<?> factory = (FactoryBean<?>) beanInstance;// 从缓存中获取我们需要的实例对象if (mbd == null && containsBeanDefinition(beanName)) {mbd = getMergedLocalBeanDefinition(beanName);}boolean synthetic = (mbd != null && mbd.isSynthetic());// 调用FactoryBean的getObject方法创建我们需要的实例对象;大家自行跟下getObjectFromFactoryBeanobject = getObjectFromFactoryBean(factory, beanName, !synthetic);}return object;
}

根据name从spring容器获取实例,如果该实例不是FactoryBean类型,则直接返回该实例,这也是我们平时用的最多的、最普通的情况;如果该实例是FactoryBean类型,而name又是以&开头,也直接返回该实例,说明我们想要的就是FactoryBean实例;如果name不是以&开头,而该实例又是FactoryBean类型,则会调用该实例的getObject()来创建我们需要的目标实例。

总结

  1. FactoryBean是BeanFactory支持的、用来暴露bean实例的接口,可以实现此接口来完成实例化过程比较复杂的bean的创建;
  2. 通过beanName从spring容器获取bean实例时,一开始获取的是beanName直接关联的bean实例,后续spring容器会根据此bean实例返回我们需要的对象实例;如果bean实例不是FactoryBean类型,则直接返回bean实例,如果bean实例是FactoryBean类型,而beanName又是以&开头,直接返回bean实例,如果bean实例是FactoryBean类型,而beanName不是以&开头,则返回bean实例的getObject()方法获取的对象实例(一般getObject中就是我们需要的实例对象的创建过程);
  3. 对于创建过程比较复杂的对象的创建,目前spring其实有很多实现方式了,而FactoryBean只是其中一种,也许我们不会采用此种方式来实现实例对象的创建,但我们需要能够看懂此种方式,知道有这种实现方式;很多第三方都沿用了此种方式,我们去追源码的时候,很容易就能碰到;
  4. 相比普通bean的创建,FactoryBean的方式会在spring容器中多存在一个FactoryBean的实例,若想获取FactoryBean实例对象,只需要在FactoryBean的beanName加&即可;

Spring拓展接口之FactoryBean相关推荐

  1. 常用的一些拓展:Spring拓展接口分门别类

    Spring框架是一个拓展性很好的框架,在平时的开发中我们也会进行一些拓展.那么来看一下常用的拓展类: ​ 编辑切换为居中 添加图片注释,不超过 140 字(可选) 这里把拓展接口分成了四大类 1. ...

  2. factorybean 代理类不能按照类型注入_Spring拓展接口之FactoryBean,我们来看看其源码实现...

    是什么 FactoryBean的源码比较简单,大家可以细读下其注释,我做了简单的如下翻译 /*** 实现此接口的bean不能用作普通bean.此bean暴露的对象是通过getObject()创建的对象 ...

  3. Spring常用的拓展接口分门别类

    Spring框架是一个拓展性很好的框架,在平时的开发中我们也会进行一些拓展.那么来看一下常用的拓展类: ​ 编辑切换为居中 添加图片注释,不超过 140 字(可选) 这里把拓展接口分成了四大类 1. ...

  4. Spring 常用的拓展接口分门别类

    Spring 框架是一个拓展性很好的框架,在平时的开发中我们也会进行一些拓展.那么来看一下常用的拓展类: ​ 编辑切换为居中 添加图片注释,不超过 140 字(可选) 这里把拓展接口分成了四大类 1. ...

  5. Spring 的微内核与FactoryBean扩展机制--转载

    作者:江南白衣 原文地址: http://www.blogjava.net/calvin/archive/2005/08/30/11099.html http://www.blogjava.net/c ...

  6. 换一种方式编写 Spring MVC 接口

    点击上方蓝色"程序猿DD",选择"设为星标" 回复"资源"获取独家整理的学习资料! 1. 前言 通常我们编写 Spring MVC 接口的范 ...

  7. java拓展接口_Java拓展接口-default关键词

    Java接口在使用过程中有两点规定: 1.接口中只能有定义方法名.方法返回类型,不能有方法的实现. 2.实现接口的类,必须实现接口中所有的方法. 例如下面的例子: //定义接口 public inte ...

  8. springbean的生命周期_spring bean生命周期(涵盖spring常用接口的载入)

    spring bean生命周期流程图: ​ ​​​ 其中包含了很多常用的接口,可以参考spring 常用接口: 下面写个例子证明下: 1.实现InitializingBean以及各个Aware接口 p ...

  9. 护网必备技能:Spring Boot 接口数据加解密 功能实现

    护网必备技能:Spring Boot 接口数据加解密 功能实现 文章目录 护网必备技能:Spring Boot 接口数据加解密 功能实现 1. 尽量少改动,不影响之前的业务逻辑: 2. 考虑到时间紧迫 ...

最新文章

  1. C# GDI+ 简单绘图 (三)
  2. 豪气!华为放话:3年培养100万AI人才!网友神回应了
  3. 如何在浏览器上跑深度学习模型?并且一行JS代码都不用写
  4. CentOS7 设置开机直接进入命令行界面
  5. R语言学习笔记-机器学习1-3章
  6. [GWCTF 2019]pyre.pyc [CISCN2018]2ex
  7. 云计算机玩大型游戏,云游戏实测点评:大部分都可以顺畅玩耍
  8. linux cant open file for writing,linux 安装rz sz lrz lsz sftp: cannot open 文件名称 to write 报错解决...
  9. NLTK自带的词干提取器
  10. sklearn 线性回归_使用sklearn库做线性回归拟合
  11. 洛谷T172100 商店-贪心
  12. “不要害怕 RAID!”
  13. 《iOS 6高级开发手册(第4版)》——1.8节使用加速度移动屏幕上的对象
  14. JavaScript数据类型之Boolean以及undefined和null(4)
  15. 【重点递归】剑指offer——面试题18:树的子结构
  16. 从0到1搭建RPC框架
  17. 如何成为一名卓越的数据科学家-桃树七剑之二:数据准备
  18. 微商爆粉2.0全自动批量加人模拟手动操作
  19. 实战演习(九)——用python分析科比生涯数据
  20. 数字化商业模式三结构:价值创造、价值交付、价值捕获

热门文章

  1. Maven的基础概念
  2. ajax判断网络中断,如何检测由于网络Ajax调用失败断开
  3. q learning简单理解_班级励志标语格言-生活其实很简单,过了今天就是明天
  4. SCOM2016 OMSDK 服务无法启动
  5. android6.0权限管理工具EasyPermissionUtil
  6. BBC:大数据带来的弊病?近因效应
  7. 让redhat5以yum方式安装软件
  8. 专访uPlane陈宏强:手机遥控固定翼飞机还是蓝海
  9. 人力资源SaaS软件“乐才Joy HR”获数百万元战略融资
  10. Hibernate5-一对多双向关联-迫切左外连接-HQL