写了一些服务器程序,监听端口,连接数据库,诸如此类的,通过命令行运行。可是程序的退出是个大问题,因为是命令行程序,所以也没有什么退出的按钮给用户去点,关闭程序时,一般是关闭控制台窗口。
最近遇到麻烦了,关闭控制台窗口以后,程序实际没有退出,端口依然被监听,和数据库的连接依然保持。。。再运行程序,报错:端口被占用
如果能知道控制台窗口被关闭的事件,那就解决我的难题了,找了好久,终于找到了这个解决办法
一个完整的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 }