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

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 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();
}


相关文章
|
14天前
|
数据采集 人工智能 Java
1天消化完Spring全家桶文档!DevDocs:一键深度解析开发文档,自动发现子URL并建立图谱
DevDocs是一款基于智能爬虫技术的开源工具,支持1-5层深度网站结构解析,能将技术文档处理时间从数周缩短至几小时,并提供Markdown/JSON格式输出与AI工具无缝集成。
79 1
1天消化完Spring全家桶文档!DevDocs:一键深度解析开发文档,自动发现子URL并建立图谱
|
15天前
|
安全 Java API
深入解析 Spring Security 配置中的 CSRF 启用与 requestMatchers 报错问题
本文深入解析了Spring Security配置中CSRF启用与`requestMatchers`报错的常见问题。针对CSRF,指出默认已启用,无需调用`enable()`,只需移除`disable()`即可恢复。对于`requestMatchers`多路径匹配报错,分析了Spring Security 6.x中方法签名的变化,并提供了三种解决方案:分次调用、自定义匹配器及降级使用`antMatchers()`。最后提醒开发者关注版本兼容性,确保升级平稳过渡。
79 2
|
1月前
|
存储 Java 文件存储
微服务——SpringBoot使用归纳——Spring Boot使用slf4j进行日志记录—— logback.xml 配置文件解析
本文解析了 `logback.xml` 配置文件的详细内容,包括日志输出格式、存储路径、控制台输出及日志级别等关键配置。通过定义 `LOG_PATTERN` 和 `FILE_PATH`,设置日志格式与存储路径;利用 `&lt;appender&gt;` 节点配置控制台和文件输出,支持日志滚动策略(如文件大小限制和保存时长);最后通过 `&lt;logger&gt;` 和 `&lt;root&gt;` 定义日志级别与输出方式。此配置适用于精细化管理日志输出,满足不同场景需求。
154 1
|
15天前
|
前端开发 安全 Java
Spring Boot 便利店销售系统项目分包设计解析
本文深入解析了基于Spring Boot的便利店销售系统分包设计,通过清晰的分层架构(表现层、业务逻辑层、数据访问层等)和模块化设计,提升了代码的可维护性、复用性和扩展性。具体分包结构包括`controller`、`service`、`repository`、`entity`、`dto`、`config`和`util`等模块,职责分明,便于团队协作与功能迭代。该设计为复杂企业级应用开发提供了实践参考。
49 0
|
15天前
|
Java 关系型数据库 MySQL
深入解析 @Transactional——Spring 事务管理的核心
本文深入解析了 Spring Boot 中 `@Transactional` 的工作机制、常见陷阱及最佳实践。作为事务管理的核心注解,`@Transactional` 确保数据库操作的原子性,避免数据不一致问题。文章通过示例讲解了其基本用法、默认回滚规则(仅未捕获的运行时异常触发回滚)、因 `try-catch` 或方法访问修饰符不当导致失效的情况,以及数据库引擎对事务的支持要求。最后总结了使用 `@Transactional` 的五大最佳实践,帮助开发者规避常见问题,提升项目稳定性与可靠性。
137 11
|
16天前
|
缓存 安全 Java
深入解析HTTP请求方法:Spring Boot实战与最佳实践
这篇博客结合了HTTP规范、Spring Boot实现和实际工程经验,通过代码示例、对比表格和架构图等方式,系统性地讲解了不同HTTP方法的应用场景和最佳实践。
81 5
|
15天前
|
安全 Java 数据安全/隐私保护
Spring Security: 深入解析 AuthenticationSuccessHandler
本文深入解析了 Spring Security 中的 `AuthenticationSuccessHandler` 接口,它用于处理用户认证成功后的逻辑。通过实现该接口,开发者可自定义页面跳转、日志记录等功能。文章详细讲解了接口方法参数及使用场景,并提供了一个根据用户角色动态跳转页面的示例。结合 Spring Security 配置,展示了如何注册自定义的成功处理器,帮助开发者灵活应对认证后的多样化需求。
48 2
|
15天前
|
前端开发 IDE Java
Spring MVC 中因导入错误的 Model 类报错问题解析
在 Spring MVC 或 Spring Boot 开发中,若导入错误的 `Model` 类(如 `ch.qos.logback.core.model.Model`),会导致无法解析 `addAttribute` 方法的错误。正确类应为 `org.springframework.ui.Model`。此问题通常因 IDE 自动导入错误类引起。解决方法包括:删除错误导入、添加正确包路径、验证依赖及清理缓存。确保代码中正确使用 Spring 提供的 `Model` 接口以实现前后端数据传递。
49 0
|
3月前
|
XML Java 开发者
Spring底层架构核心概念解析
理解 Spring 框架的核心概念对于开发和维护 Spring 应用程序至关重要。IOC 和 AOP 是其两个关键特性,通过依赖注入和面向切面编程实现了高效的模块化和松耦合设计。Spring 容器管理着 Beans 的生命周期和配置,而核心模块为各种应用场景提供了丰富的功能支持。通过全面掌握这些核心概念,开发者可以更加高效地利用 Spring 框架开发企业级应用。
124 18
|
4月前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能

推荐镜像

更多
下一篇
oss创建bucket