DefaultListableBeanFactory中创建bean之前判断该bean是否为 非抽象&&单例&&非懒加载的
!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()
AbstractBeanFactory中又判断了一次是否 单例
mbd.isSingleton()
这里是不是有重复判断了。
1、这里通过 RootBeanDefinition mbd 获取到要创建的bean的Class对象
图12、
图2这里跟入,先判断每一个BeanPostProcessors类型是否为 InstantiationAwareBeanPostProcessor,结果为true才会执行 InstantiationAwareBeanPostProcessor(BeanPostProcessor的子接口) 接口的
postProcessBeforeInstantiation(Class beanClass, String beanName)
和BeanPostProcessord的
postProcessAfterInitialization(Object bean, String beanName)
这两个方法提供子类对象重写,实现bean实例化前和后处理的逻辑。假如我们要在bean实例化前对某个bean做一些前置处理或后置处理,可以重写上述两个方法。注意:要执行后置方法的前提是前置方法有返回对象。
上面过程结束后,bean = 上述两个方法执行的结果,postProcessAfterInitialization的结果会覆盖之前的结果。这里应该把返回结果理解成代理对象比较合适。
3、
图3这里看起来是真正实例化bean的地方,跟入。
图4又看到一个创建bean的方法,继续跟入。
1)首先获取bean的Class对象。
2)发现这一步有多种创建bean的方式:
1. instantiateUsingFactoryMethod(beanName, mbd, args);这里是使用工厂方法创建,比如那些使用 @Bean注解的
2. autowireConstructor(beanName, mbd, ctors, args);看起来是使用有参构造器
3. utowireConstructor(beanName, mbd, ctors, null);未跟进
4. instantiateBean(beanName, mbd);最后是默认的实例化方法
3)
图5返回的是包装对象,1310行这里反射创建了对象。明天有空继续跟进
准备了好久的面试,能入职值得了,虽然福利不怎么好,但是可以走路去上班了。
第一个方法获取了 实例化bean的代理策略,InstantiationStrategy接口定义了实例化的方法,几个 instantiate 重载方法。
图6跟入实例化方法,从RootBeanDefinition中获取构造函数对象,第一次获取时候构造函数对象是空的,接下来通过RootBeanDefinition获取Class对象然后通过 clazz.getDeclaredConstructor() 获取到class对象的构造函数,并把构造函数保存到RootBeanDefinition中。
图7跟入BeanUtils.instantiateClass(constructorToUse),能看到最后通过构造器反射创建了对象。
图8对象创建成功后,回到图5,BeanWrapper接口包装bean实例及该bean的一些属性,跟入initBeanWrapper,initBeanWrapper方法是AbstractBeanFactory这个父类定义的,包装类设置了类型转换的对象。
下面那个注册方法跟入没看懂,注册了 PropertyEditorRegistry接口的子类,是 BeanWrapper 的爷爷接口。
PropertyEditorRegistry 和 PropertyEditorRegistrySupport 源码 - woshi123 - 博客园
走完这里,接着往图4下面的代码走。
applyMergedBeanDefinitionPostProcessors(mbd, beanType, beanName);又是一个处理后置处理器的方法,根据名字看出来是处理某个类型的后置处理器。跟进去,红框的方法对所有后置处理器进行了分类缓存,然后执行 MergedBeanDefinitionPostProcessor 这个类型的后置处理器的后置方法。执行完后将 RootBeanDefinition的标记属性改为true,表示MergedBeanDefinitionPostProcessors的后置方法执行完成。
创建完成一个bean后,还要解决bean的循环依赖问题,将高级的缓存提到低一级的缓存中。每当创建bean时,都会判断缓存中是否有实例化过的同类型的bean对象。
创建bean的步骤完了,但这才是第一部分,接下来还有初始化bean属性的内容。
网友评论