来源于一次面试中问到的问题,如何优雅的关闭Spring IOC容器,当时因为紧张,思路固化了,但同时也怪自己知识没学系统没学明白,只是知道在AbstractApplicationContext中有这么一个方法,但具体是什么却没去看过!
1.概述
每个Java应用都有一个属于自己的Runtime类实例,使应用程序能够与其运行的环境相连接,可以通过 getRuntime 方法获取/销毁当前运行状态中的一些内容或者进行内存清理。当然,应用程序不能创建自己的 Runtime 类实例。
关于Runtime类中方法的介绍,读者请自行参考 JDK Runtime类的说明
2.关于Runtime.getRuntime().addShutdownHook(shutdownHook);
在Java应用程序中可以通过添加关闭钩子,实现在程序退出时关闭资源的功能。 使用Runtime.addShutdownHook(Thread hook)向JVM添加关闭钩子,具体代码如下:
图中main方法中初始的初始的线程都会先执行完,才会去执行Message对象中的run()中的内容,执行顺序如下:
这种所谓的钩子一般会在一下几种场景中触发:
1)程序正常退出
2)使用System.exit()
3)终端使用Ctrl+C触发的中断
4)系统关闭
5)使用Kill pid命令干掉进程(并不是kill -9 pid,这个作者尝试过,具体见后文)
但在以下场景中,这个钩子不会触发的:
1.在Eclipse中直接点击Terminate关闭程序时不会触发关闭钩子的;
2.在Linux下使用kill -9也是不会触发钩子的;
3.关于Spring的关闭
大家应都知道,spring容器的核心流程都是在AbstractApplicationContext中实现的,读者可自行阅读,而关闭的流程也在这个里面,代码如下:
既然Spring实现了,我们可以直接使用其默认的,并不需要重写,但是有时候我们要实现我们自己的特定业务场景,比如某种资源的清理,也是需要手动扩展的,具体案例实现如下:
public class ShutDownHook extends Thread {
private Log logger = LogFactory.getLog(getClass());
private ConfigurableApplicationContext applicationContext;
public ShutDownHook(ConfigurableApplicationContext applicationContext ){
super();
this.applicationContext = applicationContext;
}
@Override
public void run() {
logger.info("Start clean the login info.");
//在系统快要关闭时,获取Bean实例,做一下善后工作
【略】
applicationContext.close();
logger.info("Socket server shutdown");
}
}
然后,在ShutdownHook实例化时需要传入Spring上下文,在系统关闭时调用ApplicationContext.close()方法:
public static void main(String[] args) {
ClassPathXmlApplicationContext ct =
new ClassPathXmlApplicationContext("applicationContext.xml");
Runtime.getRuntime().addShutdownHook(new ShutDownHook(ct));
//在正式部署时不需要下面的代码,这段代码仅供调试时使用
Thread thread = new ExitThread();
thread.start();
}
4.使用案例
通过使用Runtime添加钩子释放系统资源,处理一些善后的事宜这种方式在很多开源的框架中都有使用到,比如dubbo也是使用这个实现优雅停机,嗯!关于dubbo内部实现细节还得读者自行阅读