Spring容器可以管理singleton作用域的Bean的生命周期,可以调用创建、初始化、销毁等生命周期的方法。
对于prototype作用域的Bean,Spring容器只负责创建,创建后Bean的实例就交给客户端代码来管理,Spring容器不再跟踪其生命周期。
Bean的生命周期
1、根据配置情况,调用Bean的构造方法或工厂方法实例化Bean
2、利用依赖注入完成Bean中所有属性值的配置注入
3、如果Bean实现了BeanNameAware接口,则调用setBeanName()方法传入当前Bean的id。
4、如果Bean实现了BeanFactoryAware接口,则调用setBeanFactory()传入当前工厂实例的引用
5、如果Bean实现了ApplicationContextAware接口,则调用setApplicationContext()方法传入当前ApplicationContext实例的引用
6、如果BeanPostProcessor和Bean关联,则调用预初始化方法postProcessBeforeInitialzation()进行加工操作,Spring AOP即利用此实现。
7、如果Bean实现了InitializingBean接口,则调用afterPropertiesSet()方法
8、如果在配置文件中通过init-method属性指定了初始化方法,则调用初始化方法
9、如果BeanPostProcessor和Bean关联,则调用初始化方法postProcessAfterInitialization()。此时,Bean已经可以被正常使用了。
10、如果指定了作用域为singleton,则将实例放在Spring IoC的缓存池中,并触发Spring容器对该Bean的生命周期管理,如果指定作用域为prototype,则将该Bean交给调用者,由调用者管理该Bean的生命周期。
11、如果Bean实现了DisposableBean接口,则调用destory()方法销毁实例;
12、如果在配置文件中通过destory-method指定了Bean的销毁方法,则调用该方法销毁实例。
说明:
以上接口中,均只有一个方法。
并不建议让Bean实现多个接口,因为继承多个接口会使代码耦合较高。
注入依赖后的行为
Spring提供2种方式,在Bean全部属性设置完成后执行特定的行为:
- 实现InitializingBean接口,在afterPropertiesSet()方法中写代码。InitializingBean接口中只有这一个方法。
- 在xml中使用init-method属性指定要调用的方法
可以同时使用这2种方式,会先执行afterPropertiesSet(),再执行init-method属性指定的方法。
Bean销毁之前的行为
Spring提供了2种方式,在Bean销毁之前执行特定的行为:
- 实现DisposableBean接口,该接口只有destory()方法
- 在xml中用destory-method属性指定要调用的方法
可以同时使用这2种方式,会先执行destory(),再执行destory-method属性指定的方法。
说明:同时使用时,都是先执行接口中的方法,再执行xml中指定的方法。
示例
1 class Student { 2 private String name; 3 4 public Student(String name){ 5 this.name=name; 6 } 7 8 public String getName(){ 9 return name; 10 } 11 } 12 13 class Teacher implements InitializingBean, DisposableBean { 14 @Autowired 15 private Student student; 16 17 public void say(){ 18 System.out.println(student.getName()+",叫家长来一下"); 19 } 20 21 public void xmlInit(){ 22 System.out.println("正在执行xml中init-method属性指定的初始化方法"); 23 } 24 25 @Override 26 public void afterPropertiesSet() throws Exception { 27 System.out.println("正在执行InitializingBean接口中的afterPropertiesSet()方法"); 28 } 29 30 public void xmlDestory(){ 31 System.out.println("正在执行xml中destory-method属性指定的方法"); 32 } 33 34 @Override 35 public void destroy() throws Exception { 36 System.out.println("正在执行DisposableBean接口中的destory()方法"); 37 } 38 } 39 40 public class Test { 41 public static void main(String[] args) { 42 AbstractApplicationContext applicationContext=new ClassPathXmlApplicationContext("applicationContext.xml"); 43 Teacher teacher=applicationContext.getBean("teacher",Teacher.class); 44 teacher.say(); 45 applicationContext.registerShutdownHook(); 46 } 47 48 }
在基于Web的Spring容器中,系统已经提供了相应的代码,在Web应用关闭时,会自动关闭Spring容器。
在非Web的Spring的容器中,需要手动调用AbstractApplicationContext类的registerShutdownHook()方法向JVM注册一个关闭钩子,执行完前面的代码,会自动关闭Spring容器。
ApplicatioContext即Spring容器。
ApplicationContext是接口,没有实现registerShutdownHook()方法,AbstractApplicationContext是ApplicationContext的子类,实现了该方法,此处要声明为AbstractApplicationContext,不能声明为ApplicationContext。
xml中的配置:
<context:annotation-config /> <bean id="student" class="my_package.Student"> <constructor-arg value="张三" /> </bean> <bean id="teacher" class="my_package.Teacher" init-method="xmlInit" destroy-method="xmlDestory" />
运行程序,控制台输出如下:
正在执行InitializingBean接口中的afterPropertiesSet()方法 正在执行xml中init-method属性指定的初始化方法 张三,叫家长来一下 正在执行DisposableBean接口中的destory()方法 正在执行xml中destory-method属性指定的方法