spring-beans包源码阅读-3-BeanDefinition

一. BeanDefinition介绍

第二个我们来说说BeanDefinition,这个接口也是spring的核心接口之一,所有的Bean在实例化之前的各种各样的信息都记录在这个接口的实现类中。

Spring中Bean的配置来源可能很多,比如xml、@Service等注解标注的类、以及硬编码等,这时需要一个统一的类去封装Bean的配置信息,这些信息会指导Bean的初始化行为和指明Bean具有的一些特性。这个封装类就是BeanDefinition。

这个接口的位置在spring-beans模块下的org.springframework.beans.factory.config,可以看到这个接口首先在factory包下,然后在config包下。这个位置也是非常合理的,因为BeanDefinition就是在BeanFactory创建Bean实例的时候,记录这个Bean的各种信息用的,比如说Bean的Class信息、是否是单例、是否懒加载、是否是抽象Bean、有哪些属性、有哪些依赖的其他Bean、initMethod的名称等等。这些信息都是用来初始化Bean的实例的时候使用到的。顾名思义的理解就是对一个Bean的定义。

这个类可以说是BeanFactory的一个辅助类,之后可以看到,BeanFactory的各种工作都是围绕着BeanDefinition进行的,也可以说这两个接口是孪生接口,谁也离不开谁。BeanFactory无疑是spring最核心的接口,在我们正式分析BeanFactory之前,先好好看看与它关系密切的BeanDefinition接口。

二. BeanDefinition实现

GenericBeanDefinition是BeanDefinition的一个实现类,与GenericBeanDefinition平级的类还有RootBeanDefinition和ChildBeanDefinition。spring的Bean有一个特性就是可以继承,一个bean可以指明另一个bean做自己的parent,这里要注意,这种方式指明的继承关系在java虚拟机中并不存在,即,如果使用instanceof等RTTI的方式去检验这两个bean的继承关系是失效的,他们只是逻辑上的继承关系并具有一些继承的特性,比如重写和重写父bean的方法和属性等。

BeanDefinition接口的绝大多数功能都在AbstractBeanDefinition中实现:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
// AbstractBeanDefinition.java
@Nullable
private volatile Object beanClass;

@Nullable
private String scope = SCOPE_DEFAULT;

private boolean abstractFlag = false;

private boolean lazyInit = false;

private int autowireMode = AUTOWIRE_NO;

private int dependencyCheck = DEPENDENCY_CHECK_NONE;

@Nullable
private String[] dependsOn;

private boolean autowireCandidate = true;

private boolean primary = false;

private final Map<String, AutowireCandidateQualifier> qualifiers = new LinkedHashMap<>();

@Nullable
private Supplier<?> instanceSupplier;

private boolean nonPublicAccessAllowed = true;

private boolean lenientConstructorResolution = true;

@Nullable
private String factoryBeanName;

@Nullable
private String factoryMethodName;

@Nullable
private ConstructorArgumentValues constructorArgumentValues;

@Nullable
private MutablePropertyValues propertyValues;

@Nullable
private MethodOverrides methodOverrides;

@Nullable
private String initMethodName;

@Nullable
private String destroyMethodName;

private boolean enforceInitMethod = true;

private boolean enforceDestroyMethod = true;

private boolean synthetic = false;

private int role = BeanDefinition.ROLE_APPLICATION;

@Nullable
private String description;

@Nullable
private Resource resource;

没必要一个一个解读了,这些属性的作用可以说一目了然,实现中也没有很复杂的逻辑,知道BeanDefinition的作用之后,我们就可以去看spring的核心BeanFactory了~

当然BeanDefinition的实现类不止文中提到的三种,还有例如对注解支持的实现类和对Configuration类的实现类,但是目的都差不多,都是为了记录一个Bean的各种配置信息。

最后提一嘴这个AttributeAccessor接口,这个接口在Spring中也是无处不在,比如包装一个属性的类PropertyValue类就实现了这个接口。这个接口里面仅仅封装了一个Map<String, Object>,但是其作用是什么我也是不是特别清楚,在调研和学习途中,也比较了”Attribute”这个词和”Property”这两者概念的区别,但是发现google上权威的和非权威的各种解释都有矛盾的地方,甚至在java的官方词库中都没有发现”Attribute”这个词,而只有”Property”,AttributeAccessor接口本身的注释也非常的含糊其辞。所以这个地方的合理性我觉得是存疑的,如果有大佬了解这部分内容的,还请您不吝赐教。

Java官方词库:Glossary of Terms

其中对Property的解释如下:

property

Characteristics of an object that users can set, such as the color of a window.