bean的生命周期
关于bean的相关知识还有最后一个是bean的生命周期,对于生命周期,我们主要围绕着bean生命周期控
 制 来讲解:
首先理解下什么是生命周期?
- 从创建到消亡的完整过程,例如人从出生到死亡的整个过程就是一个生命周期。
bean生命周期是什么?
- bean对象从创建到销毁的整体过程。
bean生命周期控制是什么?
- 在bean创建后到销毁前做一些事情。
现在我们面临的问题是如何在bean的创建之后和销毁之前把我们需要添加的内容添加进去。
环境准备
还是老规矩,为了方便大家后期代码的阅读,我们重新搭建下环境:
- 创建一个Maven项目
- pom.xml添加依赖
- resources下添加spring的配置文件applicationContext.xml
这些步骤和前面的都一致,大家可以快速的拷贝即可,最终项目的结构如下:

(1)项目中添加BookDao、BookDaoImpl、BookService和BookServiceImpl类
1 public interface BookDao {
2     public void save();
3 }
4  
5 public class BookDaoImpl implements BookDao {
6     public void save() {
7         System.out.println("book dao save ...");
8     }
9 }
10  
11 public interface BookService {
12     public void save();
13 }
14  
15 public class BookServiceImpl implements BookService{
16     private BookDao bookDao;
17  
18     public void setBookDao(BookDao bookDao) {
19         this.bookDao = bookDao;
20     }
21  
22     public void save() {
23         System.out.println("book service save ...");
24         bookDao.save();
25     }
26 }
 
 
 
 
(2)resources下提供spring的配置文件
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
4        xsi:schemaLocation="http://www.springframework.org/schema/beans 
http://www.springframework.org/schema/beans/spring-beans.xsd">
5  
6     <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
7 </beans> 
  
(3)编写AppForLifeCycle运行类,加载Spring的IOC容器,并从中获取对应的bean对象
1 public class AppForLifeCycle {
2     public static void main( String[] args ) {
3         ApplicationContext ctx = new 
4             ClassPathXmlApplicationContext("applicationContext.xml");
5         BookDao bookDao = (BookDao) ctx.getBean("bookDao");
6         bookDao.save();
7     }
8 }生命周期设置
接下来,在上面这个环境中来为BookDao添加生命周期的控制方法,具体的控制有两个阶段:
- bean创建之后,想要添加内容,比如用来初始化需要用到资源
- bean销毁之前,想要添加内容,比如用来释放用到的资源
步骤1:添加初始化和销毁方法
针对这两个阶段,我们在BooDaoImpl类中分别添加两个方法,方法名任意
1 public class BookDaoImpl implements BookDao {
2     public void save() {
3         System.out.println("book dao save ...");
4     }
5     //表示bean初始化对应的操作
6     public void init(){
7         System.out.println("init...");
8     }
9     //表示bean销毁前对应的操作
10     public void destory(){
11         System.out.println("destory...");
12     }
13 }步骤2:配置生命周期
在配置文件添加配置,如下:
1 <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl" init-method="init" 
destroy-method="destory"/>步骤3:运行程序
运行AppForLifeCycle打印结果为:

从结果中可以看出,init方法执行了,但是destroy方法却未执行,这是为什么呢?
- Spring的IOC容器是运行在JVM中
- 运行main方法后,JVM启动,Spring加载配置文件生成IOC容器,从容器获取bean对象,然后调方法执行
- main方法执行完后,JVM退出,这个时候IOC容器中的bean还没有来得及销毁就已经结束了
- 所以没有调用对应的destroy方法
知道了出现问题的原因,具体该如何解决呢?
close关闭容器
ApplicationContext中没有close方法
需要将ApplicationContext更换成ClassPathXmlApplicationContext
1 ClassPathXmlApplicationContext ctx = new 
2     ClassPathXmlApplicationContext("applicationContext.xml");调用ctx的close()方法
1 ctx.close(); 运行程序,就能执行destroy方法的内容
 运行程序,就能执行destroy方法的内容

注册钩子关闭容器
在容器未关闭之前,提前设置好回调函数,让JVM在退出之前回调此函数来关闭容器
调用ctx的registerShutdownHook()方法
1 ctx.registerShutdownHook();注意:registerShutdownHook在ApplicationContext中也没有
运行后,查询打印结果

两种方式介绍完后,close和registerShutdownHook选哪个?
相同点:这两种都能用来关闭容器
不同点:close()是在调用的时候关闭,registerShutdownHook()是在JVM退出前调用关闭。
分析上面的实现过程,会发现添加初始化和销毁方法,即需要编码也需要配置,实现起来步骤比较多
 也比较乱。
Spring提供了两个接口来完成生命周期的控制,好处是可以不用再进行配置init-method和
destroy-method
接下来在BookServiceImpl完成这两个接口的使用:
修改BookServiceImpl类,添加两个接口InitializingBean, DisposableBean并实现接口中的
 两个方法afterPropertiesSet和destroy
1 public class BookServiceImpl implements BookService, InitializingBean, 
DisposableBean {
2     private BookDao bookDao;
3     public void setBookDao(BookDao bookDao) {
4         this.bookDao = bookDao;
5     }
6     public void save() {
7         System.out.println("book service save ...");
8         bookDao.save(); 
9     }
10     public void destroy() throws Exception {
11         System.out.println("service destroy");
12     }
13     public void afterPropertiesSet() throws Exception {
14         System.out.println("service init");
15     }
16 }重新运行AppForLifeCycle类
那第二种方式的实现,我们也介绍完了。
小细节
- 对于InitializingBean接口中的afterPropertiesSet方法,翻译过来为 属性设置之后 。
- 对于BookServiceImpl来说,bookDao是它的一个属性
- setBookDao方法是Spring的IOC容器为其注入属性的方法
- 思考:afterPropertiesSet和setBookDao谁先执行?
从方法名分析,猜想应该是setBookDao方法先执行
验证思路,在setBookDao方法中添加一句话
1 public void setBookDao(BookDao bookDao) {
2         System.out.println("set .....");
3         this.bookDao = bookDao;
4     }
5  重新运行AppForLifeCycle,打印结果如下:
 验证的结果和我们猜想的结果是一致的,所以初始化方法会在类中属性设置之后执行。
验证的结果和我们猜想的结果是一致的,所以初始化方法会在类中属性设置之后执行。
bean生命周期小结
(1)关于Spring中对bean生命周期控制提供了两种方式:
- 在配置文件中的bean标签中添加init-method和destroy-method属性
- 类实现InitializingBean与DisposableBean接口,这种方式了解下即可。
(2)对于bean的生命周期控制在bean的整个生命周期中所处的位置如下:
初始化容器
- 1.创建对象(内存分配)
- 2.执行构造方法
- 3.执行属性注入(set操作)
- 4.执行bean初始化方法
使用bean
- 1.执行业务操作
关闭/销毁容器
- 1.执行bean销毁方法
(3)关闭容器的两种方式:
ConfigurableApplicationContext是ApplicationContext的子类
- close()方法
- registerShutdownHook()方法



















