spring中所说的bean对象 与 我们自己new的对象(原始对象)是不同的;
bean对象是指spring框架创建管理的我们的对象
生命周期即:何时生,何时死
1.实例化 Instantiation:
spring通过反射机制以及工厂创建出来的原始对象;
2.属性赋值 Populate:
对对象中属性进行赋值操作
3.初始化 Ininitialization:
在这个阶段完成对原始对象的各种功能增强,如AOP生成代理对象、实现下面的接口方法等:
这些是初始化阶段可以实现的接口,以及接口中的方法:
3.1 如果 Bean 实现 BeanNameAware 执行 setBeanName
3.2 如果 Bean 实现 BeanFactoryAware 或者ApplicationContextAware 设 置工厂 setBeanFactory 或者上下文对象 setApplicationContext 对象.
3.3 如果存在类实现 BeanPostProcessor(AOP),执行postProcessBeforeInitialization
3.4 如果 Bean 实现 InitializingBean 执行 afterPropertiesSet如果配置了自己的初始化方法<bean init-method="init">
3.5 如果存在类实现 BeanPostProcessor(AOP) ,执行postProcessAfterInitialization
如下@transactional标签实现事务管理,同时这个类也实现了两个接口,实现了接口的方法:
这些都是在初始化阶段需要完成的;等这些都完成后,bean对象就算是真正意义上的创建完成了
完成后就把bean对象放入到容器中使用
4.销毁
如果 Bean 实现 DisposableBean 执行 destroy 如果配置了自己的销毁方法<bean destroy-method="customerDestroy">
指定销毁方法 customerDestroy
bean生命周期具体流程图:
Spring中的bean是线程安全的吗?
servlet是线程安全的吗?不是,因为servlet对象只创建了一个,所以属性值也就只有一个,当有多个请求(线程)访问同一个值时,就会出现线程安全问题、共享问题;
spring中的bean是否是线程安全的,要看它的作用域,即scope:
若是Singleton,即单例,那就是线程不安全,所有线程都共享一个单例实例Bean,可以使用ThreadLocal解决,为每一个线程创建一个变量副本;
若是prototype,即原型的 多例 是线程安全的,一个请求到来,创建一个对象,bean之间不存在Bean共享,自然不会出现线程安全问题。
bean又分为 有状态bean 和 无状态bean
1.有状态:可以用来存储数据
2.无状态:不会存储数据,如Service层,Dao层只是方法调用
如果单例 Bean,是一个无状态 Bean,也就是线程中的操作不会对 Bean 的成员执行查询以外的操作,那么这个单例 Bean 是线程安全的。比如 Spring mvc 的 Controller、Service、Dao 等,这些 Bean 大多是无状态的,只关注于方法本身。
但是如果单例Bean 是有状态的,那就不能保证线程安全, 那就需要开发人员自己来进行线程安全的保证,最简单的办法1.就是改变 bean 的作用域把 "singleton"改为’‘protopyte’‘, 这样每次请求 Bean 就相当于是 new Bean() 这样就可以保证线程的安全了。 2.就是使用Threadlocal线程池存储变量,为每个线程创建一个变量副本
controller、service 和 dao 层本身并不是线程安全的,只是如果只是调用里面的方法,而且多线程调用一个实例的方法,会在内存中复制变量,这是自己的线程的工作内存,是安全的。
Bean循环依赖
java中循环依赖场景:
很简单,A类关联了B类,B类同时也关联了A类;
如果是纯粹的java语言,即不考虑spring,自己new对象时,是没有任何问题的,因为对象之间相互关联是很常见的事,我们直接将关联在内的对象初始化赋值为null,然后相互赋值即可,如下。
Spring产生循环依赖问题
但是在spring管理bean时,在自动注入时,如果两个类相互之间关联,那么会出现注入一个对象时,另一个对象还没有初始化完成(bean对象需要经历初始化过程)。总之一个对象并不会简单new出来,而是会经过一系列的Bean的声明周期,就是因为Bean的生命周期所以才会出现循环依赖问题。
如何产生循环依赖问题:主要是:A创建时→需要B----B去创建→又需要A
Spring解决循环依赖
spring提供了 三级缓存机制来解决这个问题:
在 DefaultSingletonBeanRegistry 类中定义了 3 个 Map 对象充当缓存;
singletonObjects: 一级缓存,用于保存实例化、注入、初始化完成的 bean实例,即已经完全初始化的bean对象;
earlySingletonObjects:二级缓存,用于保存实例化完成的 半成品bean 实例,即就是刚创建出的半成品对象,未经初始化,提前曝露对象给需要的地方。
singletonFactories: 三级缓存,用于保存创建 bean 对象的工厂
具体过程:
当A中需要B时,那么先从一级缓存中找B,没有,然后到二级缓存中找B,若也没有,则去三级缓存中,找到B对应的工厂,创建出B对象(原始对象,半成品),把B对象存入二级缓存中,如下:
主要通过三级缓存解决Bean循环依赖问题:
出现问题的根本原因是实例化和初始化是分开处理的。