以前在开发时只知道依靠数据库事务来保证程序关闭时数据的完整性。

但有些时候一个业务上要求的原子操作,不一定只包括数据库,比如外部接口或者消息队列。此时数据库事务就无能为力了。

这时我们可以依靠java提供的一个工具方法:java.lang.Runtime.addShutdownHook(Thread hook)

addShutdownHook方法可以加入一个钩子,在程序退出时触发该钩子。

(退出是指ctrl+c或者kill -15,但如果用kill -9 那是没办法的,具体有关kill的signal机制有篇大牛的文章《Linux 信号signal处理机制》

钩子做什么操作都可以,甚至可以循环检查某个线程的状态,直到业务线程正常退出,再结束钩子程序就可以保证业务线程的完整性

例子程序如下:

实例程序在执行过程中按下ctrl -c或者 kill -15,由于钩子程序的循环检测,能够保证线程执行完毕后,程序才关闭。

  1. /**
  2. * Created by IntelliJ IDEA.
  3. * User: Luo
  4. * Date: 13-7-11
  5. * Time: 下午3:12
  6. */
  7.  
  8. public class TestShutdownHook {
  9.  
  10. /**
  11. * 测试线程,用于模拟一个原子操作
  12. */
  13. private static class TaskThread extends Thread {
  14. @Override
  15. public void run() {
  16. System.out.println("thread begin ...");
  17. TestShutdownHook.sleep(1000);
  18. System.out.println("task 1 ok ...");
  19. TestShutdownHook.sleep(1000);
  20. System.out.println("task 2 ok ...");
  21. TestShutdownHook.sleep(1000);
  22. System.out.println("task 3 ok ...");
  23. TestShutdownHook.sleep(1000);
  24. System.out.println("task 4 ok ...");
  25. TestShutdownHook.sleep(1000);
  26. System.out.println("task 5 ok ...");
  27.  
  28. System.out.println("thread end\n\n");
  29. }
  30. }
  31.  
  32. /**
  33. * 注册hook程序,保证线程能够完整执行
  34. * @param checkThread
  35. */
  36. private static void addShutdownHook(final Thread checkThread) {
  37. //为了保证TaskThread不在中途退出,添加ShutdownHook
  38. Runtime.getRuntime().addShutdownHook(new Thread() {
  39. public void run() {
  40. System.out.println("收到关闭信号,hook起动,开始检测线程状态 ...");
  41. //不断检测一次执行状态,如果线程一直没有执行完毕,超时后,放弃等待 \
  42. for (int i = 0; i < 100; i++) {
  43. if (checkThread.getState() == State.TERMINATED) {
  44. System.out.println("检测到线程执行完毕,退出hook");
  45. return;
  46. }
  47. TestShutdownHook.sleep(100);
  48. }
  49. System.out.println("检测超时,放弃等待,退出hook,此时线程会被强制关闭");
  50. }
  51. });
  52. }
  53.  
  54. private static void sleep(long millis) {
  55. try {
  56. Thread.sleep(millis);
  57. } catch (InterruptedException e) {
  58. e.printStackTrace();
  59. }
  60. }
  61.  
  62. public static void main(String[] args) throws InterruptedException {
  63. final TaskThread taskThread = new TaskThread();
  64. //为了保证TaskThread不在中途退出,添加ShutdownHook
  65. addShutdownHook(taskThread);
  66. //执行TaskThread
  67. taskThread.start();
  68. }
  69.  
  70. }

  

java 保证程序安全退出的更多相关文章

  1. 【Java】一个简单的Java应用程序

    简单记录,Java 核心技术卷I 基础知识(原书第10 版) 一个简单的Java应用程序"Hello, World!" Hello, World! Goodbye,World! 一 ...

  2. 利用 java.lang.Runtime.addShutdownHook() 钩子程序,保证java程序安全退出

    以前在开发时只知道依靠数据库事务来保证程序关闭时数据的完整性. 但有些时候一个业务上要求的原子操作,不一定只包括数据库,比如外部接口或者消息队列.此时数据库事务就无能为力了. 这时我们可以依靠java ...

  3. 初步探究java中程序退出、GC垃圾回收时,socket tcp连接的行为

    初步探究java中程序退出.GC垃圾回收时,socket tcp连接的行为 今天在项目开发中需要用到socket tcp连接相关(作为tcp客户端),在思考中发觉需要理清socket主动.被动关闭时发 ...

  4. java基础(死循环退出选项)

    java程序中为了程序正常运行,需要给无限循环加入一个退出选项,保证程序的可执行性. import java.util.Scanner; public class { public static vo ...

  5. 使用 Acegi 保护 Java 应用程序

    第 1 部分: 架构概览和安全过滤器 Acegi Security System 是一种功能强大并易于使用的替代性方案,使您不必再为 Java 企业应用程序编写大量的安全代码.虽然它专门针对使用 Sp ...

  6. [转]基于java的程序OutOfMemory问题的解决及Xms/Xmx/Xss的解释和应用

    长期以来一直都是做java应用的开发,所使用的开发工具基本上也都是基于java的,经常用的有eclipse, netbeans, ant, maven, cruisecontrol, tomcat, ...

  7. 在并发Java应用程序中检测可见性错误

    了解什么是可见性错误,为什么会发生,以及如何在并发Java应用程序中查找难以捉摸的可见性错误.这些问题你可能也遇到过,当在优锐课学习了一段时间后,我对这些问题有了一定见解,写下这篇文章和大家分享. 检 ...

  8. 软件性能测试分析与调优实践之路-Java应用程序的性能分析与调优-手稿节选

    Java编程语言自从诞生起,就成为了一门非常流行的编程语言,覆盖了互联网.安卓应用.后端应用.大数据等很多技术领域,因此Java应用程序的性能分析和调优也是一门非常重要的课题.Java应用程序的性能直 ...

  9. 《Java应用程序(Application)》

    在编写Java应用程序(Application)时可以这样: 1,定义包名. 2, 导入相关的包. 3, 定义一个类. 4,定义相关变量. 5,定义构造函数.(在构造函数内调用init()方法和add ...

随机推荐

  1. 101210-450789-147200(可以激活Xshell5,而且可以升级) 亲测可用 只能用于xshell5

    101210-450789-147200(可以激活Xshell5,而且可以升级) 亲测可用 只能用于xshell5

  2. Python学习—数据库篇之pymysql

    一.pymysql简介 对于Python操作MySQL主要使用两种方式: 原生模块 pymsql ORM框架 SQLAchemy pymsql是Python中操作MySQL的模块,其使用方法和MySQ ...

  3. 《Orange‘s》FAT12文件系统

    FAT12 层次 扇区(Sector):磁盘上的最小数据单元 簇(Cluster):一个或多个扇区 分区(Partition):通常指整个文件系统 引导扇区 引导扇区是整块软盘的第0个扇区,在这个扇区 ...

  4. mysql 2pc理解

  5. 矢量水听器 近场 远场 --------------------常规波束形成,MVDR的比较

    摘自<水平线阵的反卷积常规波束形成>,IRONMAN--------------------------------------------------- 在常规的波束形成中,是将阵列上的 ...

  6. vue学习笔记(WebStorm安装)

    慕课网:https://www.imooc.com/video/18553 一.前往官网下载:https://www.jetbrains.com/webstorm/download/#section= ...

  7. K3精益版给物料添加属性,并在BOM中新增字段引用该属性

    1.给物料新增属性 打开“系统--基础资料--公共资料--核算项目管理”,然后双击物料,弹出核算项目类别-修改对话框.再点新增按钮: 输入你想新增字段的类型,长度,想要放置的位置. 相关属性里面选的是 ...

  8. Spring的xml配置文件中约束的必要性 找不到元素 'beans' 的声明

    今天在复习Spring MVC框架的时候,只知道xml配置文件中的约束有规范书写格式的作用,所以在配置HandlerMapping对象信息的时候没有加入约束信息之后进行测试,没有遇到问题.后来在配置S ...

  9. vue中部分api解释 ($nextTick)

    1:this.$nextTick(function(){ }) 传如的参数是一个函数 这个API主要是获取dom元素 为什么需要这个api,在vue框架开发中,更新dom是一个异步操作,如果更新完do ...

  10. 密码与安全新技术专题之AI与密码

    20189217 2018-2019-2 <密码与安全新技术专题>第五周作业 课程:<密码与安全新技术专题> 班级: 1892 姓名: 李熹桥 学号:20189214 上课教师 ...