在上一篇博客中,我们已经对bean的加载跟着源代码完整的走了一遍,代码读起来很轻松,主要还是Spring的编程风格,它把每段代码分割成一个一个的小逻辑,使得每个逻辑处理起来都不是很负责,非常有利于我们学习。接下来,我们继续对bean实例的创建进行阅读。
Spring启动之后,如果我们不主动从里面获取bean的实例,Spring是不会主动创建实例的,所以实例的创建流程是从下面这段代码开始的
Person person = (Person) beanFactory.getBean("person");
虽然只有简单的一句,但是他后面的逻辑确是很复杂的,我们开始。
public Object getBean(String name) throws BeansException {
return doGetBean(name, null, null, false);
}
同样符合我们发现的规律,干活的永远都是do开头的方法,这就类似于被戴了帽子吧!
protected <T> T doGetBean(final String name, @Nullable final Class<T> requiredType,
@Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
//首先将用户传入的名字变成在IOC存储的beanName
final String beanName = transformedBeanName(name);
Object bean;
// 先按照用户要到的bean是单例来查找
Object sharedInstance = getSingleton(beanName);
//如果能从IOC容器的缓存中查到,那么这个bean一定是单例并且已经被加载过了
if (sharedInstance != null && args == null) {
...
//由于用户获得的bean可能是继承了FactroyBean的bean,所以还要进行一些验证和处理
bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
}
else {
// 我们假设在循环引用当中
if (isPrototypeCurrentlyInCreation(beanName)) {
throw new BeanCurrentlyInCreationException(beanName);
}
...
try {
//获得一个合并的beanDefinition,其实就是复制了一个beanDefinition并且对其中缺少的属性进行填充
final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
checkMergedBeanDefinition(mbd, beanName, args);
//确保初始化当前bean所依赖的bean都初始化了
String[] dependsOn = mbd.getDependsOn();
if (dependsOn != null) {
for (String dep : dependsOn) {
if (isDependent(beanName, dep)) {
throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Circular depends-on relationship between '" + beanName + "' and '" + dep + "'");
}
registerDependentBean(dep, beanName);
getBean(dep);
}
}
// 创建单例
if (mbd.isSingleton()) {
//创建单例的具体方法
sharedInstance = getSingleton(beanName, () -> {
try {
return createBean(beanName, mbd, args);
}
catch (BeansException ex) {
destroySingleton(beanName);
throw ex;
}
});
bean =