循环依赖
首先是A类里面有个B属性,
B类里面有个A属性
spring每次创建bean时,都会去一级缓存中去找,没有找到则创建,这里是先实例化,
然后去将实例化好的bean加入到三级缓存中,value是一个lambda表达式
然后实例化好之后再去初始化,初始化的过程中是需要对属性进行填充的,此时有一个引用类型的属性,由于没有这个属性所对应的·bean还没有进行创建,所以我们就要去创建这个bean,相当又重头开始了会去
首先创建A对象,先去一级缓存中找,没有则进行实例化,实例化后判断是否存在一级缓存里,很显然,不存在,就 将该半成品的bean放入到三级缓存中,
然后对开始进行初始化,开始对属性B进行填充, 也是先去一级缓存中找B这个bean,没有则进行创建,实例化B,实例化好好后,判断是否在一级缓存中存在,很显然不存在,则加入到三级缓存中,然后开始进行属性填充,填充A属性,判断A属性是否在创建中,如果是,则去二级缓存中去找,发现没有,则去三级缓存中去找,此时是有的,(此时三级缓存有两个A,B都放进去了),
然后将得到的三级缓存中的A放入到二级缓存中,此时放入到二级缓存的A对象依然是个半成品
B对象既然通过三级缓存拿到了A对象,那么就一系列的初始化过程就完成了,完成了就会将B对象的成品对象放入到一级缓存,将三级缓存和二级缓存的B对象删除
那接下来就是方法的一系列返回了,返回到了A对象的初始化过程。此时A对象去一级缓存中找B对象,发现存在,则就注入属性,初始化好后,将A对象放入到一级缓存,删除23级缓存中的A
bean的生命周期
1、解析类得到BeanDefinition
2、如果有多个构造方法,则要推断构造方法
3、确定好构造方法后,进行实例化得到一个对象
4、对对象中的加了@Autowired注解的属性进行属性填充
5、回调Aware方法,比如BeanNameAware,BeanFactoryAware
6、调用BeanPostProcessor的初始化前的方法(postProcessBeforeInitialization)
7、调用初始化方法
8、调用BeanPostProcessor的初始化后的方法,在这里会进行AOP(postProcessAfterInitialization)
9、如果当前创建的bean是单例的则会把bean放入单例池
10、使用bean
11、Spring容器关闭时调用DisposableBean中destory()方法
AWare:存在的意义:方便通过spring中的bean对象来获取对应容器中的相关属性
代理就在这里实现的BeanPostProcessor
如果需要进行aop,则会进行aop代理,如果不需要,则直接返回bean,它是通过bean是否实现接口来判断使用那种代理方式
其中我们还要知道有一个类是BeanDefination 是描述的一个类的定义信息,
但是我们可以去修改它,通过实现BeanFactoryPostProcessor 来去修改
可以修改它最终被生成的类是什么类,以及这个类被初始化是使用的是哪个构造方法,
以及注入的时候,不使用@Autoeired等注入属性的注解,也可以实现注入,就是因为可以去修改它的注入模型
但在实际开发中一般是使用不到的,但有在使用,如扩展的与mybatis整合,opfeign整合等
这里这个类定义信息还涉及到springboot的自动装配,自动装配装配的就是类的定义信息
将对象放入到ioc容器中,不使用@Service @Repostory @Component @controller等也可以实现
有三种实现方式
第一种
首先定义一个类,只有一个构造,因为我们知道我们创建的默认是单例对象,是立即加载,当spring启动是就会将我们的bean加入到ioc容器中去,那必定会初始化执行构造方法。所以我们只需要启动程序,有打印这几个字就可以了。
public class Service {
Service(){
System.out.println("初始化");
}
}
在定义这么一个配置类就可以了
@Configuration
@Import(value={Service.class})
public class MyConfig {
}
当然这是在springboot的环境下,当然spring的注解容器也是可以的
第二种(注册器的方式)
这个类不做改变
public class Service {
Service(){
System.out.println("初始化");
}
}
新建一个类去实现一个 导入bean定义注册器
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar{
@Override
public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
//构建一个Service的bean定义
RootBeanDefinition rootBeanDefinition = new RootBeanDefinition(Service.class);
//注册到容器中去 ,并且设置这个bean的名称
registry.registerBeanDefinition("service",rootBeanDefinition);
}
}
MyConfig稍作修改,value修改为上面的实现了bean定义注册器的类
@Configuration
@Import(value={MyImportBeanDefinitionRegistrar.class})
public class MyConfig {
}
第三种(选择器的方式)
这个类依然不做改变,也没有任何注解
public class Service {
Service(){
System.out.println("初始化");
}
}
新建一个类去实现选择器
public class MyImportSelector implements ImportSelector{
@Override
public String[] selectImports(AnnotationMetadata annotationMetadata) {
return new String[]{"com.example.springbootstart.service.Service"};
}
//要指定类的全限定类名,可以有多个
}
MyConfig的@Import的value值要修改一下
@Configuration
@Import(value={MyImportSelector.class})
public class MyConfig {
}
其中第三种是springboot用来自动装配的方式
第二章=种则是spring用来整合mybatis,springcloud,openfeign等组件的