浅谈Spring Bean对象创建过程

一、引言

本文以BeanFactory为切入点,浅谈Spring是如何创建Bean的。

二、何为BeanFactory

当谈到Spring时,我们首先联想到的是IOC、AOP两大特性,这种高逼格的东西,让一些java初学者望而却步,但其实它本质上就是个"Bean容器"。而容器中的Bean对象就是由BeanFactory生产的,BeanFactory通过工厂模式实现对Bean对象的管理从而隐藏了对象的具体实现细节。下面我们看看BeanFactory是如何创建Bean的

三、实例化Bean对象

Bean在实例化之前,必须是在Bean容器启动之后。所以我们分两个阶段介绍:

  1. Bean容器的启动阶段;
  2. Bean对象实例化阶段;

启动阶段

  1. 首先SpringBoot会读取Bean的xml配置文件,或者扫描相关注解,转换成一个BeanDefinition对象,其中保存了该bean的各种信息:
  • beanClass: bean的class属性
  • scope:bean是否单例
  • abstractFlag: 该bean是否抽象
  • lazyInit: 是否延迟初始化
  • autowireMode:保存是否自动装配
  • dependencyCheck:保存是否坚持依赖
  • dependsOn:保存该bean依赖于哪些bean(这些bean必须在初始化时提前初始化)
  • constructorArgumentValues:保存通过构造函数注入的依赖
  • propertyValues:保存通过setter方法注入的依赖
  • factoryBeanNamefactoryMethodName:用于factorybean,即工厂类型的bean
  • initMethodNamedestroyMethodName:分别对应bean的init-method和destroy-method属性
  1. 然后通过BeanDefinitionRegistry将这些bean注册到BeanFactory中:
java 复制代码
public interface BeanDefinitionRegistry extends AliasRegistry {
    void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)throws BeanDefinitionStoreException;
    void removeBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
    BeanDefinition getBeanDefinition(String beanName) throws NoSuchBeanDefinitionException;
    boolean containsBeanDefinition(String beanName);
    String[] getBeanDefinitionNames();
    int getBeanDefinitionCount();
    boolean isBeanNameInUse(String beanName);
}

BeanFactory的实现类,需要实现BeanDefinitionRegistry 接口:

java 复制代码
@SuppressWarnings("serial")
public class DefaultListableBeanFactory extends AbstractAutowireCapableBeanFactory
        implements ConfigurableListableBeanFactory, BeanDefinitionRegistry, Serializable {
    /** Map of bean definition objects, keyed by bean name */
    private final Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap<String, BeanDefinition>(64);
        @Override
    public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
            throws BeanDefinitionStoreException {
        // ... ...       this.beanDefinitionMap.put(beanName, beanDefinition);
       // ... ...
    }

在这里BeanDefinition被注册到了 DefaultListableBeanFactory, 保存在它的一个ConcurrentHashMap中。

实例化阶段

实例化阶段主要是通过反射或者CGLIB对bean进行实例化,在这个阶段Spring又给我们暴露了很多的扩展点:

  1. 各种的Aware接口,比如 BeanFactoryAwareMessageSourceAwareApplicationContextAware
  2. BeanPostProcessor接口: 实现了 BeanPostProcessor 接口的bean,在实例化bean时Spring会帮我们调用接口中的方法:
    • 在 Bean 初始化前后进行额外处理:通过重写 BeanPostProcessor 接口的方法,可以在目标 Bean 的初始化方法(eg:InitializingBeanafterPropertiesSet)调用之前(postProcessBeforeInitialization)和之后(postProcessAfterInitialization)执行自定义的处理操作。
  3. InitializingBean接口:实现了InitializingBean接口的bean,在实例化bean时Spring会帮我们调用afterPropertiesSet方法。
  4. DisposableBean接口:实现了BeanPostProcessor接口的bean,在该bean死亡时Spring会帮我们调用destroy方法。

补充:注解@PostConstruct@PreDestroy 也能达到 InitializingBean接口 和 DisposableBean接口的效果。

四. 总结

spring容器接管了bean的实例化,不仅仅是通过依赖注入达到了松耦合的效果,同时给我们提供了各种的扩展接口,来在bean的生命周期的各个时期插入我们自己的代码:

0) BeanFactoryPostProcessor接口(在容器启动阶段)

1) 各种的Aware接口

2) BeanPostProcessor接口

3) InitializingBean接口(@PostConstruct, init-method)

4) DisposableBean接口(@PreDestroy, destory-method)

相关推荐
Mr.朱鹏20 分钟前
操作002:HelloWorld
java·后端·spring·rabbitmq·maven·intellij-idea·java-rabbitmq
编程洪同学2 小时前
Spring Boot 中实现自定义注解记录接口日志功能
android·java·spring boot·后端
小小药2 小时前
009-spring-bean的实例化流程
java·数据库·spring
GraduationDesign2 小时前
基于SpringBoot的蜗牛兼职网的设计与实现
java·spring boot·后端
颜淡慕潇2 小时前
【K8S问题系列 | 20 】K8S如何删除异常对象(Pod、Namespace、PV、PVC)
后端·云原生·容器·kubernetes
customer083 小时前
【开源免费】基于SpringBoot+Vue.JS安康旅游网站(JAVA毕业设计)
java·vue.js·spring boot·后端·kafka·开源·旅游
搬码后生仔4 小时前
将 ASP.NET Core 应用程序的日志保存到 D 盘的文件中 (如 Serilog)
后端·asp.net
Suwg2094 小时前
《手写Mybatis渐进式源码实践》实践笔记(第七章 SQL执行器的创建和使用)
java·数据库·笔记·后端·sql·mybatis·模板方法模式
herogus丶4 小时前
【Spring AI】Spring AI Alibaba的简单使用
java·人工智能·spring·ai
凡人的AI工具箱5 小时前
每天40分玩转Django:Django文件上传
开发语言·数据库·后端·python·django