• Java开发学习(五)----bean的生命周期


    一、什么是生命周期

    • 首先理解下什么是生命周期?

      • 从创建到消亡的完整过程,例如人从出生到死亡的整个过程就是一个生命周期。

    • bean生命周期是什么?

      • bean对象从创建到销毁的整体过程。

    • bean生命周期控制是什么?

      • 在bean创建后到销毁前做一些事情。

    二、环境准备

    环境搭建:

    • 创建一个Maven项目

    • pom.xml添加依赖

    • resources下添加spring的配置文件applicationContext.xml

    最终项目的结构如下:

    (1)项目中添加BookDao、BookDaoImpl、BookService和BookServiceImpl类

    public interface BookDao {
        public void save();
    }
    ​
    public class BookDaoImpl implements BookDao {
        public void save() {
            System.out.println("book dao save ...");
        }
    }
    ​
    public interface BookService {
        public void save();
    }
    ​
    public class BookServiceImpl implements BookService{
        private BookDao bookDao;
    ​
        public void setBookDao(BookDao bookDao) {
            this.bookDao = bookDao;
        }
    ​
        public void save() {
            System.out.println("book service save ...");
            bookDao.save();
        }
    }

    (2)resources下提供spring的配置文件

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"
           xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    ​
        <bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
    </beans>

    (3)编写AppForLifeCycle运行类,加载Spring的IOC容器,并从中获取对应的bean对象

    public class AppForLifeCycle {
        public static void main( String[] args ) {
            ApplicationContext ctx = new 
                ClassPathXmlApplicationContext("applicationContext.xml");
            BookDao bookDao = (BookDao) ctx.getBean("bookDao");
            bookDao.save();
        }
    }

    三、生命周期设置

    接下来,在上面这个环境中来为BookDao添加生命周期的控制方法,具体的控制有两个阶段:

    • bean创建之后,想要添加内容,比如用来初始化需要用到资源

    • bean销毁之前,想要添加内容,比如用来释放用到的资源

    步骤1:添加初始化和销毁方法

    针对这两个阶段,我们在BooDaoImpl类中分别添加两个方法,方法名任意

    public class BookDaoImpl implements BookDao {
        public void save() {
            System.out.println("book dao save ...");
        }
        //表示bean初始化对应的操作
        public void init(){
            System.out.println("init...");
        }
        //表示bean销毁前对应的操作
        public void destory(){
            System.out.println("destory...");
        }
    }
    步骤2:配置生命周期

    在配置文件添加配置,如下:

    <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

      ClassPathXmlApplicationContext ctx = new 
          ClassPathXmlApplicationContext("applicationContext.xml");
    • 调用ctx的close()方法

      ctx.close();
    • 运行程序,就能执行destroy方法的内容

    这种方式比较暴力,直接就关闭掉容器了,接下来介绍另一种方式,

    五、注册钩子关闭容器

    • 在容器未关闭之前,提前设置好回调函数,让JVM在退出之前回调此函数来关闭容器

    • 调用ctx的registerShutdownHook()方法

    • 是全局变量,方法放置位置无所谓

      ctx.registerShutdownHook();

      注意:registerShutdownHook在ApplicationContext中也没有

    • 运行后,查询打印结果

    两种方式介绍完后,close和registerShutdownHook选哪个?

    相同点:这两种都能用来关闭容器

    不同点:close()是在调用的时候关闭,registerShutdownHook()是在JVM退出前调用关闭。

    分析上面的实现过程,会发现添加初始化和销毁方法,即需要编码也需要配置,实现起来步骤比较多也比较乱。

    Spring提供了两个接口来完成生命周期的控制,好处是可以不用再进行配置init-methoddestroy-method

    接下来在BookServiceImpl完成这两个接口的使用:

    修改BookServiceImpl类,添加两个接口InitializingBeanDisposableBean并实现接口中的两个方法afterPropertiesSetdestroy

    public class BookServiceImpl implements BookService, InitializingBean, DisposableBean {
        private BookDao bookDao;
        public void setBookDao(BookDao bookDao) {
            this.bookDao = bookDao;
        }
        public void save() {
            System.out.println("book service save ...");
            bookDao.save(); 
        }
        public void destroy() throws Exception {
            System.out.println("service destroy");
        }
        public void afterPropertiesSet() throws Exception {
            System.out.println("service init");
        }
    }

    重新运行AppForLifeCycle类

    那第二种方式的实现,我们也介绍完了。

    小细节

    • 对于InitializingBean接口中的afterPropertiesSet方法,翻译过来为属性设置之后

    • 对于BookServiceImpl来说,bookDao是它的一个属性

    • setBookDao方法是Spring的IOC容器为其注入属性的方法

    • 思考:afterPropertiesSet和setBookDao谁先执行?

      • 从方法名分析,猜想应该是setBookDao方法先执行

      • 验证思路,在setBookDao方法中添加一句话

        public void setBookDao(BookDao bookDao) {
                System.out.println("set .....");
                this.bookDao = bookDao;
            }
        ​
      • 重新运行AppForLifeCycle,打印结果如下:

        验证的结果和我们猜想的结果是一致的,所以初始化方法会在类中属性设置之后执行。

    六、bean生命周期总结

    (1)关于Spring中对bean生命周期控制提供了两种方式:

    • 在配置文件中的bean标签中添加init-methoddestroy-method属性

    • 类实现InitializingBeanDisposableBean接口。

    (2)对于bean的生命周期控制在bean的整个生命周期中所处的位置如下:

    • 初始化容器

      • 1.创建对象(内存分配)

      • 2.执行构造方法

      • 3.执行属性注入(set操作)

      • 4.执行bean初始化方法

    • 使用bean

      • 1.执行业务操作

    • 关闭/销毁容器

      • 1.执行bean销毁方法

    (3)关闭容器的两种方式:

    • ConfigurableApplicationContext是ApplicationContext的子类

      • close()方法

      • registerShutdownHook()方法

     

     



    如果您觉得阅读本文对您有帮助,请点一下“推荐”按钮,您的“推荐”将是我最大的写作动力!欢迎各位转载,但是未经作者本人同意,转载文章之后必须在文章页面明显位置给出作者和原文连接,否则保留追究法律责任的权利。
  • 相关阅读:
    快速掌握keepalived
    C—数据的储存(下)
    详细全面的postman接口测试实战教程
    自动填充字段值
    babel的配置执行顺序
    【Android笔记02】Android目录结构及相关文件介绍
    MySQL34道例题
    【C/C++】STL——容器适配器:stack和queue的使用及模拟实现
    beamManagement(一)idle初始接入过程
    基于OXC的光电联动全光网组网方案研究与实践
  • 原文地址:https://www.cnblogs.com/xiaoyh/p/16311026.html