上一章说,想要模拟完成Mybatis的Mapper自动注入,而这个自动注入使用了BeanDefinition,所以我们要先了解一下这个东西。才能看懂下一章的代码。
简而言之的话,是Bean的建模对象。
对Java有了解的话,一定都知道一句话叫做:万物皆对象。
那么对于java来说,Bean也是一个对象,既然是对象,那么就肯定可以抽象出来,通过一定的结构进行存储。那么这个抽象出来的结构就是BeanDefinition。
BeanDefinition里面记录了Bean对象的所有的属性,比如构造方法、构造方法参数、基础属性、是否单例、是否懒加载、等等等等。
BeanDefinition在spring真正的使用中是一个公共的接口,框架中实现了好几个不同类型的BeanDefinition。我们这里先列举几个公共的属性初步了解一下。
● ConstructorArgumentValues constructorArgumentValues:如果这个对象中有值,spring会把他作为构造方法的参数进行填充
● String scope = SCOPE_DEFAULT :默认单例
● String[] dependsOn:是否依赖了别的bean
● Object beanClass:这个BeanDefinition所描述的类
● MutablePropertyValues propertyValues:如果这个对象中有值,spring会把他作为setter方法填充
如果简单来说一下对象的实例化过程的话:假设磁盘中有N个.java文件,首先会把这些java文件编辑成.class文件,程序启动后,JVM会把这些class文件的内容load到内存中,当遇到new关键字的时候,就根据类的模板信息进行实例化。
但是呢,在Spring框架真正的运行中,情况还是比较复杂的。
- 启动Spring容器,扫描那些加了注解的类(加了注解才是要交给Spring容器管理的),将这些类存入一个集合中。
- 遍历这个集合,解析对象,给每一个类都生成一个BeanDefinition
- 把初始化好的BeanDefinition存入到Map中
- 调用所有的BeanFactoryPostProcessor(这是Bean生命周期中必然经历的,后面会详细讲)
- spring在实例化之前会对BeanDefinition对象进行解析验证,看看当前这个对象是否需要现在就进行实例化。
- 如果一切正常的话,按照流程继续走,会直接实例化出来一个对象,完成自动注入,最后放入单例池中等待调用。
- 如果我们自己实现了BeanFactoryPostProcessor,对其中的BeanDefinitionMap做了修改扩展之类的,那么就不会走第六部,而是先执行这个第7步。如下
@Component
public class X {
public void get() {
System.out.println("X Bean");
}
}
@Component
public class Y {
public void get() {
System.out.println("Y Bean");
}
}
public class Z {
public void get() {
System.out.println("Z Bean");
}
}
@Component
public class ZBeanFactoryPostProcessor implements BeanFactoryPostProcessor {
@Override
public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
AbstractBeanDefinition x = (AbstractBeanDefinition)beanFactory.getBeanDefinition("x");
x.setBeanClass(Z.class);
}
}
@Test
public void TestBeanFactory(){
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
context.scan("com.spring.beanFactory");
context.refresh();
context.getBean(Y.class).get();
context.getBean(Z.class).get();
context.getBean(X.class).get();
}
//执行结果
Y Bean
Z Bean
org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'com.spring.beanFactory.X' available
也就是说,在BeanDefinition中的对象被实例化之前,我们是可以修改这个对象的,那么就给我们注入Mapper提供了操作的空间。