Spring5源码 - 08 BeanFactory和FactoryBean 源码解析 & 使用场景

简介: Spring5源码 - 08 BeanFactory和FactoryBean 源码解析 & 使用场景

20200914143739548.png

BeanFactory VS FactoryBean


首先明确一下,这两个东西是完全不同的两个东西 ,不要混淆。

BeanFactory 是Spring Framework的 顶级核心接口 , 没有这个接口,就没有Bean的产生。

FactoryBean也是一个接口,是一个特殊的Bean , 实现了FactoryBean 接口的Bean,原来的Bean将会被隐藏,而是由FactoryBean 的getObjecct方法返回最终的Bean 。 简单工厂模式 。


如果需要获取FactoryBean实例本身,需要 & 。


可以把FactoryBean理解为4S店,改装你原来的Bean。


FactoryBean VS 普通Bean


FactoryBean和普通Bean不同,其返回的对象不是指定类的一个实例,而是该FactoryBean的getObject方法所返回的对象。创建出来的对象是否属于单例由isSingleton中的返回决定。


20200923213115667.png


演示

20200923202215557.png


【Bean1】

package com.artisan.factoryBean;
import org.springframework.stereotype.Component;
import javax.annotation.PostConstruct;
@Component
public class Bean1 {
  @PostConstruct
  public void init(){
    System.out.println("bean1 create ");
  }
}


【SpecialBeanFB 实现了 FactoryBean接口 】 用于装饰Bean

package com.artisan.factoryBean;
import org.springframework.beans.factory.FactoryBean;
import org.springframework.stereotype.Component;
import javax.annotation.PostConstruct;
@Component
public class SpecialBeanFB implements FactoryBean {
  @PostConstruct
  public void init(){
    System.out.println("SpecialBean as Factory Bean create ");
  }
  @Override
  public Object getObject()  {
    return new Bean2();
  }
  @Override
  public Class<?> getObjectType() {
    return Bean2.class;
  }
  @Override
  public boolean isSingleton() {
    return true;
  }
}


如果SpecialBeanFB这个FactoryBean上的Component注解 增加了 name ,例如@Component("aaa") 其实是个getObject对象起的名字,而不是本身这个FactoryBean实例


当我们需要获取FactoryBean实例本身而不是它所产生的bean,要使用&符号


比如这里的 id为”specialBeanFB”的FactoryBean :


  • 调用getBean(“specialBeanFB”)将返回FactoryBean产生的bean
  • 调用getBean("&specialBeanFB")将返回FactoryBean它本身的实例

【配置类】

package com.artisan.factoryBean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
@Configuration
@ComponentScan("com.artisan.factoryBean")
public class FBConfig {
}


【测试类】

package com.artisan.factoryBean;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
public class FactoryBeanTest {
  public static void main(String[] args) throws Exception {
    // 实例化Spring Bean 容器
    AnnotationConfigApplicationContext ac = new AnnotationConfigApplicationContext(FBConfig.class);
    // 从bean容器中读取普通的bean
    System.out.println(ac.getBean(Bean1.class));
    System.out.println(ac.getBean("bean1"));
    System.out.println("===========");
    // 从bean容器中读取FactoryBean接口修饰的Bean
    System.out.println(ac.getBean(SpecialBeanFB.class).getObject().getClass().getName());
    System.out.println(ac.getBean("specialBeanFB"));
    System.out.println("===========");
    // 通过 &  获取 FactoryBean本身的bean对象
    System.out.println(ac.getBean("&specialBeanFB"));
    System.out.println("===========");
    // 如果SpecialBeanFB这个FactoryBean上的Component注解 增加了 name ,例如@Component("aaa")
    // 其实是个getObject对象起的名字,而不是本身这个FactoryBean实例
//    System.out.println(ac.getBean("aaa"));
//    System.out.println(ac.getBean("&aaa"));
  }
}


【测试结果】


2020092320471181.png


源码



20200923205059428.png

在实例化Bean的方法中

AbstractApplicationContext # refresh() 
    ---------- finishBeanFactoryInitialization(beanFactory) 
       ----- DefaultListableBeanFactory#preInstantiateSingletons


在Spring容器启动阶段,会调用到refresh()方法,在refresh()中调用了finishBeanFactoryInitialization()方法,最终会调用到beanFactory.preInstantiateSingletons()方法

public void preInstantiateSingletons() throws BeansException {
  // 从容器中获取到所有的beanName
  List<String> beanNames = new ArrayList<>(this.beanDefinitionNames);
  for (String beanName : beanNames) {
    // 合并Bean   ScannedGenericBeanDefinition AnnotatedGenericBeanDefinition 类型的Bean 等等 都要合并为 RootBeanDefinition 比较复杂,知道即可
    RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName);
    // 创建Bean的条件校验
    if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) {
      // 在此处会根据beanName判断bean是不是一个FactoryBean,实现了FactoryBean接口的bean,会返回true 
      if (isFactoryBean(beanName)) {
        // 然后通过getBean()方法去获取或者创建单例对象
        // 注意:在此处为beanName拼接了一个前缀:FACTORY_BEAN_PREFIX  是一个常量字符串,即:&
        // 所以在此时容器启动阶段,对于specialBeanFB,应该是:getBean("&specialBeanFB")
        Object bean = getBean(FACTORY_BEAN_PREFIX + beanName);
        // 下面这一段逻辑,是判断是否需要在容器启动阶段,就去实例化getObject()返回的对象,即是否调用FactoryBean的getObject()方法
        if (bean instanceof FactoryBean) {
          final FactoryBean<?> factory = (FactoryBean<?>) bean;
          boolean isEagerInit;
          if (System.getSecurityManager() != null && factory instanceof SmartFactoryBean) {
            isEagerInit = AccessController.doPrivileged((PrivilegedAction<Boolean>)
                    ((SmartFactoryBean<?>) factory)::isEagerInit,
                getAccessControlContext());
          }
          else {
            isEagerInit = (factory instanceof SmartFactoryBean &&
                ((SmartFactoryBean<?>) factory).isEagerInit());
          }
          if (isEagerInit) {
            getBean(beanName);
          }
        }
      }
    }
  }
}


使用场景


比如 MyBatis3 提供 mybatis-spring项目中的 org.mybatis.spring.SqlSessionFactoryBean

SqlSessionFactoryBean 看名字的结尾FactoryBean


public class SqlSessionFactoryBean implements FactoryBean<SqlSessionFactory>, InitializingBean, ApplicationListener<ApplicationEvent> {
  // ...省略其他代码
  public SqlSessionFactory getObject() throws Exception {
  if (this.sqlSessionFactory == null) {
    afterPropertiesSet();
  }
  return this.sqlSessionFactory;
  }
}


sqlSessionFactory是SqlSessionFactoryBean的一个属性,它的赋值是在通过回调afterPropertiesSet()方法进行的。 因为SqlSessionFactoryBean实现了InitializingBean接口,所以在Spring初始化Bean的时候,能回调afterPropertiesSet()方法 .

public void afterPropertiesSet() throws Exception {
    // buildSqlSessionFactory()方法会根据mybatis的配置进行初始化。
  this.sqlSessionFactory = buildSqlSessionFactory();
}


相关文章
|
前端开发 Java 物联网
智慧班牌源码,采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署
智慧班牌系统是一款基于信息化与物联网技术的校园管理工具,集成电子屏显示、人脸识别及数据交互功能,实现班级信息展示、智能考勤与家校互通。系统采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署与私有化定制。核心功能涵盖信息发布、考勤管理、教务处理及数据分析,助力校园文化建设与教学优化。其综合性和可扩展性有效打破数据孤岛,提升交互体验并降低管理成本,适用于日常教学、考试管理和应急场景,为智慧校园建设提供全面解决方案。
748 70
|
11月前
|
设计模式 Java 开发者
如何快速上手【Spring AOP】?从动态代理到源码剖析(下篇)
Spring AOP的实现本质上依赖于代理模式这一经典设计模式。代理模式通过引入代理对象作为目标对象的中间层,实现了对目标对象访问的控制与增强,其核心价值在于解耦核心业务逻辑与横切关注点。在框架设计中,这种模式广泛用于实现功能扩展(如远程调用、延迟加载)、行为拦截(如权限校验、异常处理)等场景,为系统提供了更高的灵活性和可维护性。
1416 0
|
算法 测试技术 C语言
深入理解HTTP/2:nghttp2库源码解析及客户端实现示例
通过解析nghttp2库的源码和实现一个简单的HTTP/2客户端示例,本文详细介绍了HTTP/2的关键特性和nghttp2的核心实现。了解这些内容可以帮助开发者更好地理解HTTP/2协议,提高Web应用的性能和用户体验。对于实际开发中的应用,可以根据需要进一步优化和扩展代码,以满足具体需求。
1504 29
|
前端开发 数据安全/隐私保护 CDN
二次元聚合短视频解析去水印系统源码
二次元聚合短视频解析去水印系统源码
588 4
|
JavaScript 算法 前端开发
JS数组操作方法全景图,全网最全构建完整知识网络!js数组操作方法全集(实现筛选转换、随机排序洗牌算法、复杂数据处理统计等情景详解,附大量源码和易错点解析)
这些方法提供了对数组的全面操作,包括搜索、遍历、转换和聚合等。通过分为原地操作方法、非原地操作方法和其他方法便于您理解和记忆,并熟悉他们各自的使用方法与使用范围。详细的案例与进阶使用,方便您理解数组操作的底层原理。链式调用的几个案例,让您玩转数组操作。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
负载均衡 JavaScript 前端开发
分片上传技术全解析:原理、优势与应用(含简单实现源码)
分片上传通过将大文件分割成多个小的片段或块,然后并行或顺序地上传这些片段,从而提高上传效率和可靠性,特别适用于大文件的上传场景,尤其是在网络环境不佳时,分片上传能有效提高上传体验。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
Java Spring 容器
Spring中BeanFactory和FactoryBean的区别?
一位工作了4年的小伙伴,去京东面试被问到这样一个问题,Spring中的BeanFactory和FactoryBean有什么区别?因为没有看过源码,当时就感觉这是一个文字游戏,感觉没什么区别? 那今天,我就给大家来聊清楚。另外,往期面试题解析中配套的文档我已经准备好,想获得的可以在我的煮叶简介中找到。好了,我们先来看BeanFactory。
259 0
|
XML Java 数据格式
Spring中BeanFactory和FactoryBean详解
Spring中BeanFactory和FactoryBean详解
981 1
|
Java Spring 容器
【Spring源码】 BeanFactory和FactoryBean是什么?
面试官:“看过Spring源码吧,简单说说Spring中BeanFactory和FactoryBean的区别是什么?”
17667 6
【Spring源码】 BeanFactory和FactoryBean是什么?

推荐镜像

更多
  • DNS