25

I'm interested in different approaches to gracefully shutting down a Java command line program. Sending a kill signal is not an option.

I can think of a few different approaches.

  1. Open a port and wait for a connection. When one is made, gracefully shutdown.
  2. Watch for a file to be created, then shutdown.
  3. Read some input from the terminal, such as "execute shutdown".

The third one is not ideal, since there is often program output pumped to the screen. The first one takes too much effort (I'm lazy). Do most programmers use the second option? If not, what else is possible/elegant/simple?

rgettman
  • 176,041
  • 30
  • 275
  • 357
  • You mean for a process to kill your java command line program? – Tom Jun 22 '09 at 18:22
  • 3
    Why is sending the kill signal not an option? As mentioned in answers below, you can catch and handle the kill signal gracefully with shutdown hooks. – toluju Jun 22 '09 at 18:33

7 Answers7

18

you can try something like this:

Runtime.getRuntime().addShutdownHook(new Thread() {
    public void run() { /*
       my shutdown code here
    */ }
 });

edit:

the shutdown hook will not perform the shutting down of the app. instead, it gives the developer a way to perform any clean-up that he/she wishes at shutdown.

from the JavaDoc for Runtime (a good read if you are planning to use this method):

A shutdown hook is simply an initialized but unstarted thread. When the virtual machine begins its shutdown sequence it will start all registered shutdown hooks in some unspecified order and let them run concurrently. When all the hooks have finished it will then run all uninvoked finalizers if finalization-on-exit has been enabled. Finally, the virtual machine will halt. ...

akf
  • 38,619
  • 8
  • 86
  • 96
8

you could try to use Runtime.getRuntime().addShutdownHook() that should satisfy your requisite. In this way you can register an hook to do cleanups, in order to perfom a gracefull shutdown.

EDIT

http://java.sun.com/j2se/1.4.2/docs/api/java/lang/Runtime.html#addShutdownHook(java.lang.Thread) public void addShutdownHook(Thread hook) Registers a new virtual-machine shutdown hook. The Java virtual machine shuts down in response to two kinds of events:

  • The program exits normally, when the last non-daemon thread exits or when the exit (equivalently, System.exit) method is invoked, or
  • The virtual machine is terminated in response to a user interrupt, such as typing ^C, or a system-wide event, such as user logoff or system shutdown.
dfa
  • 114,442
  • 31
  • 189
  • 228
6

The benefit of the second option - checking for a file - over the first - listening on a port - is that you have some possibility of security.

You can set the permissions on the directory where the file is created so that only appropriate users can close the program. If you listen on a port any user can connect to it.

David Webb
  • 190,537
  • 57
  • 313
  • 299
  • You could limit the incoming calls based on the client IP address. For example allow connections only from localhost. You only exit when it is the proper IP address, otherwise close the connection and continue listening for a new one. – akarnokd Jun 22 '09 at 18:38
  • @kd304 - Limiting IP addresses still allows any user on local system to shut down the process. By setting permissions on the directory you can only allow a specific user control of the process. – David Webb Jun 22 '09 at 18:52
4

If you wanted to go with the socket version, it is very simple to implement. Here's the code:

ServerSocket server = new ServerSocket(8080);
System.out.println("Socket listening!");
server.accept();
System.out.println("Connection received!");

You could easily embed this code in a separate thread that you start with your program, and then have it modify global state to initiate shutdown.

toluju
  • 4,097
  • 2
  • 23
  • 27
2

The first two option is simple to implement. You could also use some JMX stuff (I don't know much about that). Tomcat uses the first approach and I applied 1 and 2 in two of my projects.

akarnokd
  • 69,132
  • 14
  • 157
  • 192
2

Consider having a JMX component. Then you can attach with JConsole either locally or over the network, and communicate with your component. Then the component can shut down the program properly.

With Java 6 u 10 or later, you can do the same with JVisualVM.

Thorbjørn Ravn Andersen
  • 73,784
  • 33
  • 194
  • 347
-1

I would suggest to use the shutdown hook. It will allow your program do be controlled using standard OS tools. It also does not need any additional access to external resources (disk, ports, whatever).

Jeroen van Bergen
  • 1,046
  • 6
  • 7