提到Spring,老是让人第一时间想起IOC容器,而IOC容器的顶层核心接口就是咱们的BeanFactory
,若是可以理解BeanFactory
的体系结构想必能让咱们对Spring总体脉络有更加清晰的认知,因此,本文的探究方向主要为如下几点:java
BeanFactory
的体系结构是怎样的?Bean
的元信息从何而来?BeanFactory
生产Bean
的过程是怎样的?咱们先来看看有哪些子类实现了它吧spring
其中ApplicationContext
这一块已在上篇文章有详细说明,而DefaultListableBeanFactory
这个底层实现类便理所固然的成为了咱们这次探究的出发点,为了让咱们有个更好的观感,如下是纯粹的BeanFactory
UML图:设计模式
咱们能够看到DefaultListableBeanFactory
实现的接口有:spring-mvc
SingletonBeanRegistry
: 定义了对单例缓存池相关的操做,如将bean注册到单例缓存池中ConfigurableBeanFactory
: 可配置的BeanFactory
,定义了各类各样的配置能力,如bean的做用域,bean的classLoader,添加bean的后置处理器,设置bean的建立状态,销毁bean等等AutowireCapableBeanFactory
: 能进行自动装配的BeanFactory
,这多是咱们最为熟悉的BeanFactory
,定义了自动装配的类型(byName/byType),createBean
, autowireBean
, 自动装配属性, populateBean, initializeBean, 对于与bean生命周期相关的方法都将在这里体现ListableBeanFactory
: 对BeanFactory
的加强,定义了一系列根据beanType获取bean或者beanName的方法ConfigurableListableBeanFactory
: 对ConfigurableBeanFactory
的加强,定义了忽略bean的类型、缓存bean定义、预实例化单例bean等方法BeanDefinitionRegistry
: bean定义注册器,定义了与bean定义相关的方法若是说以上的接口体现了DefaultListableBeanFactory
具有的功能,那么它所继承的一系列类就是这些功能的实现:缓存
DefaultSingletonBeanRegistry
: 单例bean注册器,定义了三级缓存,其实就是三个Map属性FactoryBeanRegistrySupport
: 提供对FactoryBean
的支持AbstractBeanFactory
: 实现了一系列操做IOC容器的功能,但最终的createBean依旧交由子类AbstractAutowireCapableBeanFactory
完成AbstractAutowireCapableBeanFactory
: 实现了建立bean的功能,全部与建立bean的相关的功能都在这里DefaultListableBeanFactory
: 在以上父类的功能基础上实现了ConfigurableBeanFactory
和BeanDefinitionRegistry
接口,定义了一些存放Bean定义相关信息的Map看到这里,想必对
DefaultListableBeanFactory
已经有一个大体的了解了,那么问题来啦,咱们应该怎么从容器中获取一个bean呢?是否是只要经过BeanDefinitionRegistry
注册一个bean定义,再经过AutowireCapableBeanFactory
去createBean就完成了呢?就像下面这样:mybatis
DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); RootBeanDefinition beanDefinition = new RootBeanDefinition(Wheel.class); beanFactory.registerBeanDefinition("wheel",beanDefinition); beanFactory.getBean("wheel", Wheel.class);
Bean
的元信息从何而来?咱们如今已经知道DefaultListableBeanFactory
的大体功能了,咱们发现当咱们想要建立一个Bean
的时候,老是离不开一个名词:Bean定义,那么这个Bean定义到底是什么呢?mvc
BeanDefinition
实际上是一个接口,并非一个具体的类,咱们也能够看一下它的UML图:app
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传源码分析
能够发现这里使用了模板方法的设计模式扩展了许多的子类,其中咱们最经常使用的为RootBeanDefinition
,它主要包含的属性以下:
post
咱们向容器中注册的Bean定义的信息大概就是如此,当BeanFactory
生产Bean
时,即可以经过beanClass
清楚的知道Bean
的类是什么,做用域是怎样,是否懒加载,init方法是什么等等等等
咦,若是一个最简单的bean,好像能够直接经过反射就搞定了耶~
具体结构已经清楚了,那咱们来看一下注册过程吧
public static void main(String[] args) { //建立一个DefaultListableBeanFactory实例 DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory(); //建立一个BeanDefinition RootBeanDefinition beanDefinition = new RootBeanDefinition(Wheel.class); //将BeanDefinition注册到容器中 beanFactory.registerBeanDefinition("wheel",beanDefinition); } public static class Wheel { }
public RootBeanDefinition(@Nullable Class<?> beanClass) { //初始化父类 super(); //将beanClass赋值给this.BeanClass setBeanClass(beanClass); }
初始化父类
//将其中一部分属性赋予默认值 autowireCandidate = true; primary = false; protected AbstractBeanDefinition() { this(null, null); } protected AbstractBeanDefinition(@Nullable ConstructorArgumentValues cargs, @Nullable MutablePropertyValues pvs) { this.constructorArgumentValues = cargs; this.propertyValues = pvs; }
//除去校验逻辑,注册时只作了这两步操做 this.beanDefinitionMap.put(beanName, beanDefinition); this.beanDefinitionNames.add(beanName);
看到这里,你们伙可能会充满疑问?what? 就这样?没有填充属性的步骤吗?嘿嘿,BeanFactory
是一个纯粹的工厂,只负责生产Bean
,是没有装配(设计)BeanDefinition
的功能的,专业的事仍是交给专业的人作,设计的事情仍是交由ApplicationContext
完成的。
那在ApplicationContext
中是怎么完成一个BeanDefinition
的呢?还记得预启动将配置类注册到容器中时有这样一段代码吗?如下代码为AnnotatedBeanDefinitionReader#doRegisterBean
中的摘要部分:
<T> void doRegisterBean(Class<T> annotatedClass, @Nullable Supplier<T> instanceSupplier, @Nullable String name, @Nullable Class<? extends Annotation>[] qualifiers, BeanDefinitionCustomizer... definitionCustomizers) { //.......代码 //处理普通的bean定义注解,@Lazy @Primary @DependsOn @Role @Description AnnotationConfigUtils.processCommonDefinitionAnnotations(abd); //......代码 }
而非配置的Bean
是经过在预启动时注册的配置类后置处理器ConfigurationClassPostProcessor#processConfigBeanDefinitions
中完成的,如下代码为ClassPathBeanDefinitionScanner#doScan
中的摘要部分,详细调用链将在后面的文章进行说明
//传入咱们配置类的包路径 protected Set<BeanDefinitionHolder> doScan(String... basePackages) { Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>(); for (String basePackage : basePackages) { //寻找到全部标识了@Component注解的BeanDefinition Set<BeanDefinition> candidates = findCandidateComponents(basePackage); for (BeanDefinition candidate : candidates) { //....省略代码 if (candidate instanceof AbstractBeanDefinition) { //处理BeanDefinition postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName); } if (candidate instanceof AnnotatedBeanDefinition) { //处理普通的bean定义注解,@Lazy @Primary @DependsOn @Role @Description AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate); } //...省略代码 //将BeanDefinition注册到容器中 registerBeanDefinition(definitionHolder, this.registry); } }
处理BeanDefinition
protected void postProcessBeanDefinition(AbstractBeanDefinition beanDefinition, String beanName) { //设置默认值 beanDefinition.applyDefaults(this.beanDefinitionDefaults); //这里默认为空 if (this.autowireCandidatePatterns != null) { beanDefinition.setAutowireCandidate(PatternMatchUtils.simpleMatch(this.autowireCandidatePatterns, beanName)); } }
设置默认值
public void applyDefaults(BeanDefinitionDefaults defaults) { //默认为null Boolean lazyInit = defaults.getLazyInit(); if (lazyInit != null) { setLazyInit(lazyInit); } //默认为0 setAutowireMode(defaults.getAutowireMode()); //默认为0 setDependencyCheck(defaults.getDependencyCheck()); //默认为null setInitMethodName(defaults.getInitMethodName()); setEnforceInitMethod(false); //默认为null setDestroyMethodName(defaults.getDestroyMethodName()); setEnforceDestroyMethod(false); }
处理普通的bean定义注解
public static void processCommonDefinitionAnnotations(AnnotatedBeanDefinition abd) { processCommonDefinitionAnnotations(abd, abd.getMetadata()); } static void processCommonDefinitionAnnotations(AnnotatedBeanDefinition abd, AnnotatedTypeMetadata metadata) { //从元数据中取出该注解的属性列表,不为空说明有标识该注解 AnnotationAttributes lazy = attributesFor(metadata, Lazy.class); if (lazy != null) { abd.setLazyInit(lazy.getBoolean("value")); } else if (abd.getMetadata() != metadata) { lazy = attributesFor(abd.getMetadata(), Lazy.class); if (lazy != null) { abd.setLazyInit(lazy.getBoolean("value")); } } //判断元数据中是否有该注解 if (metadata.isAnnotated(Primary.class.getName())) { abd.setPrimary(true); } AnnotationAttributes dependsOn = attributesFor(metadata, DependsOn.class); if (dependsOn != null) { abd.setDependsOn(dependsOn.getStringArray("value")); } AnnotationAttributes role = attributesFor(metadata, Role.class); if (role != null) { abd.setRole(role.getNumber("value").intValue()); } AnnotationAttributes description = attributesFor(metadata, Description.class); if (description != null) { abd.setDescription(description.getString("value")); } }
attributesFor(metadata, Lazy.class)
static AnnotationAttributes attributesFor(AnnotatedTypeMetadata metadata, Class<?> annotationClass) { return attributesFor(metadata, annotationClass.getName()); } static AnnotationAttributes attributesFor(AnnotatedTypeMetadata metadata, String annotationClassName) { //metadata为beanClass的注解元数据,存放了该类所配置的全部注解 //annotationClassName为须要寻找的注解名称 return AnnotationAttributes.fromMap(metadata.getAnnotationAttributes(annotationClassName, false)); } default Map<String, Object> getAnnotationAttributes(String annotationName, boolean classValuesAsString) { //遍历元数据中的全部注解 MergedAnnotation<Annotation> annotation = getAnnotations().get(annotationName, null, MergedAnnotationSelectors.firstDirectlyDeclared()); //不存在则返回null,不然返回一个map if (!annotation.isPresent()) { return null; } return annotation.asAnnotationAttributes(Adapt.values(classValuesAsString, true)); }
以上为扫描@Component注解类进行解析元数据填充属性的逻辑,在配置类中以@Bean方式注册的BeanDefinition填充属性过程在ConfigurationClassBeanDefinitionReader.loadBeanDefinitions(configClasses)
BeanFactory
生产Bean
的过程是怎样的?如今,咱们已经知道了一个BeanDefinition
的具体结构以及是如何产生并注册到BeanFactory
的,那么BeanFactory
又是如何使用它生产Bean
的呢?如下附上createBean
的粗略流程图,具体细节将放在IOC容器启动流程中详细说明
那么关于
BeanFactory
的相关内容就到这里了,但愿你们有所收获,下篇将正式进入Spring IOC容器的启动流程!
准备花个30天时间,系统的来整理一下我对spring源码的认识:
暂定
追更,可扫下面的二维码,分享纯粹为了乐趣,也有一种成就感吧,笔者这篇文章先就到这