如何在java控制台程序退出时执行特定的处理

时间:2024-03-11 15:50:12
写了一些服务器程序,监听端口,连接数据库,诸如此类的,通过命令行运行。可是程序的退出是个大问题,因为是命令行程序,所以也没有什么退出的按钮给用户去点,关闭程序时,一般是关闭控制台窗口。

最近遇到麻烦了,关闭控制台窗口以后,程序实际没有退出,端口依然被监听,和数据库的连接依然保持。。。再运行程序,报错:端口被占用

 

如果能知道控制台窗口被关闭的事件,那就解决我的难题了,找了好久,终于找到了这个解决办法

 

一个完整的Java应用程序,通常至少要有一个应用程序的结束点。对于一般程序来说,系统开发者根据需要和个人的偏好,会在程序结束位置,通过添加System.exit(0),或System.out(-1),来结束程序,或不加这些指令,让程序自然运行到结束。

如:下列典型代码

/**
 * This application is to demo how an applcation end
 */
 public class Test {

 public Test() {
 }

 public static void main(String[] args) {

 Test test1 = new Test();
 System.out.println("hello world");
 System.exit(0);//也可以不写这句代码,让程序自然结束。
}

}

对于简单的应用系统,我们直接可以在System.exit(0)代码执行前,添加需要在应用程序退出前需要完成的工作,如:关闭网络连接,关闭数据库连接等。
然而,对于比较复杂的多线程应用,线程运行的状态较复杂,我们就很难预料程序何时结束,如何能在应用程序结束事件到来时,处理我们要做的工作呢?这就用到了Java对应用程序的退出的事件出处理机制。
对当前应用程序对象的获得,Java通过Runtime静态方法:Runtime.getRuntime()通过Runtime的 void addShutdownHook(Thread hook) 法向Java虚拟机注册一个shutdown钩子事件,这样一旦程序结束事件到来时,就运行线程hook,我们在实际应用时候,只要将程序需要完成之前做的一些工作直接通过线程hook来完成。具体演示代码如下

 1 /*****************************************************************************
 2 * 本程序仅演示,如何在Java应用程序中添加系统退出事件处理机制
 3 *****************************************************************************/
 4 import java.util.*;
 5 import java.io.*;
 6 /**
 7 * This application is used to demo how to hook the event of an application
 8 */
 9 
10 public class Untitled1 {
11 
12 public Untitled1() { 
13    doShutDownWork(); 
14 }
15 
16 /*************************************************************************** 
17    * This is the right work that will do before the system shutdown
18       * 这里为了演示,为应用程序的退出增加了一个事件处理,
19    * 当应用程序退出时候,将程序退出的日期写入 d:\t.log文件
20    **************************************************************************/
21 
22 private void doShutDownWork() {
23    Runtime.getRuntime().addShutdownHook(new Thread() {
24     public void run() {
25      try {
26       FileWriter fw = new FileWriter("d:\\t.log");
27       System.out.println("Im going to end");
28       fw.write("the application ended! " + (new Date()).toString());
29       fw.close();
30      } catch (IOException ex) {
31      }    
32     }  
33    });
34 
35 }
36 
37 /****************************************************
38    * 这是程序的入口,仅为演示,方法中的代码无关紧要
39    ***************************************************/
40 
41 public static void main(String[] args) {
42 
43    Untitled1 untitled11 = new Untitled1();  
44    long s = System.currentTimeMillis();  
45    for (int i = 0; i < 1000000000; i++) {  
46    //在这里增添您需要处理代码
47    }
48    long se = System.currentTimeMillis();
49    System.out.println(se - s);
50 }
51 
52 }