博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java Daemon Control
阅读量:6630 次
发布时间:2019-06-25

本文共 7305 字,大约阅读时间需要 24 分钟。

博客分类:

Java Daemon Control

王福强(Darren.Wang)

2010-07-27


Table of Contents

1. Old Days Solutions
2. Alternatives Available Today
3. My Choice
A. 参考文档

As to desktop or normal Java applications, we can easily know when we should shutdown the application or not, because users have explicit ways to do this, for example, for a Swing Application, usually a "CLOSE" menu or tool-bar item will be available, or directly click the "X" icon on the left/right top of the window. But for a Java application that will be run as a server process(which don't need interactive behavior), what we do?

1. Old Days Solutions

Directly kill -9 ? Of course, that's a way, but that's too brutal.

A Java process that will be run as a server process usually will be sent to OS's background to run, that's called daemon on Unix and service on Windows. A Simple way to control the life-cycle of a Java daemon is to start a loop and wait for user input, like this:

Java代码  
  1. BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));  
  2. String line = null;  
  3. do {  
  4.     line = reader.readLine();  
  5.     if (line != null && line.equals("quit")) {  
  6.         break// and exit gracefully  
  7.     }  
  8. }  
  9. while (true);  
  10.   
  11. // clean up and exit  
BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));String line = null;do {    line = reader.readLine();    if (line != null && line.equals("quit")) {        break; // and exit gracefully    }}while (true);// clean up and exit
 

This solution is better than just start an infinite loop and do nothing, because the latter solution will occupy all of your CPU's power(I have seen such a stupid solution and it does exist). But this is still not a best one, although users can interactive with it, the process itself have no way to notify the process to exit. That's, you can control it from outside, but can't do it from inside.

Another mostly used solution is, start a TCP server socket and listen control requests, when termination control request is received, a loop based on control flag will break and exit. This is similar to above solution, just with another communication channel. [ ]

A common pattern for both solution can be found,that's, set up a control flag to start a loop with, and then wait for other channels to change the control flag. This can be formulated as:

Java代码  
  1. boolean running = true;  
  2.   
  3. while(running)  
  4. {  
  5.     // do Sth.  
  6. }  
  7. // exit  
boolean running = true;while(running){	// do Sth.}// exit
 

As to how to change the control flag, there are two solutions presented, can you find more?

2. Alternatives Available Today

have their pros and cons, today more solutions are available for us.

The first one is . It provides a Java daemon solution with native Jsvc and Procrun support.

The second and third are and , The former is a commerce solution now, and the latter is an open source one which has some works based on the former(there is a possibility that I misunderstand its introduction).

Other Solutions are , Start-Stop-Daemon, or , but I don't get patience to read their document. If you are interested in them,follow the links I provide or google them.

3. My Choice

I choose to combine shell and sun.misc.Signal and sun.misc.SignalHandler to achieve Java daemon, because they are much simpler to me to understand them and use them. The Shell will take care of running-in-background stuff:

Java代码  
  1. #!/bin/sh  
  2. java -cp your_class_path com.domain.main_class <&- &  
  3. pid=$!  
  4. echo ${pid} > mydaemon.pid  
#!/bin/shjava -cp your_class_path com.domain.main_class <&- &pid=$!echo ${pid} > mydaemon.pid
 

And sun.misc.Signal and sun.misc.SignalHandler will take care of controlling the life-cycle of the program.

About sun.misc.Signal and sun.misc.SignalHandler , you can find more information at , here I just simply introduce how to use them together to achieve asynchronous communication between processes or just internally in a same process.

The concept of sun.misc.Signal and sun.misc.SignalHandler is simple:

  • Signal is the signal that u will send to SignalHandler to process, so you can create a Signal just like instantiate a simple value object:

    Java代码  
    1. Signal sig = new Signal("USR1");  
    Signal sig = new Signal("USR1");
     

     

    The signal names you pass to Signal conform a pattern, that's, remove the prefix "SIG" from the name of the standard signals that's used by JVM [ ] . For example, if you want to send SIGINT , then you create Signal instance with name of INT ; If you want to send SIGTERM , you create Signal instance with name of TERM :

    Java代码  
    1. Signal interactiveSignal = new Signal("INT");  
    2. Signal terminationSignal = new Signal("TERM");  
    Signal interactiveSignal = new Signal("INT");Signal terminationSignal = new Signal("TERM");
     

     

    Fucking Simple, right?

    After you have a Signal, you can send it out by using Signal class's raise method:

    Java代码  
    1. Signal.raise(sig);   
    2. Signal.raise(interactiveSignal);   
    3. Signal.raise(terminationSignal);   
    Signal.raise(sig); Signal.raise(interactiveSignal); Signal.raise(terminationSignal);
     

     

     

  • As the name indicates, SignalHandler will take the responsibility of handling the Signal s.

    You implements your own signal handlers by implementing the SignalHandler interface. It has only one method:

    Java代码  
    1. public class MySigHandler implements SignalHandler {  
    2.   
    3.     public void handle(Signal sig) {  
    4.         // ...  
    5.     }  
    6. }  
    public class MySigHandler implements SignalHandler {    public void handle(Signal sig) {		// ...    }}
     

     

    It's fucking simple too.

After you get both Signal and SignalHandler of your own, you should link them together to make it work. This is by Signal class's static method handle :

Java代码  
  1. MySigHandler sigHandler = new MySigHandler();  
  2. Signal.handle(sig, sigHandler);  
  3. Signal.handle(interactiveSignal, sigHandler);  
  4. Signal.handle(terminationSignal, sigHandler);  
MySigHandler sigHandler = new MySigHandler();Signal.handle(sig, sigHandler);Signal.handle(interactiveSignal, sigHandler);Signal.handle(terminationSignal, sigHandler);
 

Now as long as you add them to your java programs and send proper signals to it, the pairs of Signal and SignalHandler will work for you.

You have 2 ways to send signals to your program's process:

  1. Use Signal.raise() internally.  This can help to coordinate application's internal state and help to control the life-cycle internally. For example, as long as internal worker thread dies, it can send out a signal, when signal hander finds that all other the worker threads die, it can change the control flag of the whole process and terminate it gracefully.

  2. Send Signal from other processes.  directly send out supported OS signal via shell scripts:

    Java代码  
    1. kill -s SIGUSR1 <pid of the process>  
    kill -s SIGUSR1 
     

     

    combining the pid you get in before shell, this works perfectly.

The only cons to use Signal and SignalHandler is, they are both restricted API which are not guaranteed.

A. 参考文档

Revelations on Java signal handling and termination . http://www.ibm.com/developerworks/java/library/i-signalhandling/ .

基于OS信号实现Java异步通知 . http://kenwublog.com/java-asynchronous-notify-based-on-signal .

Java Daemon . http://barelyenough.org/blog/2005/03/java-daemon/ .


[ ] To use this solution, you'd better add authentication to your control service so that others with malicious purpose will not hurt you.

[ ] you can find these standard signals at .

转载地址:http://kgwvo.baihongyu.com/

你可能感兴趣的文章
Java并发编程的艺术
查看>>
批量分发ssh公钥证书
查看>>
iOS encrypt Md5, Sha1,Base64
查看>>
git 常用命令
查看>>
Android系统启动流程(四)Launcher启动过程与系统启动流程
查看>>
jquery增,删,改一个html标签的class名字
查看>>
缓存技术
查看>>
怎么样将自己开发的Android应用程序编译到系统Image中
查看>>
Android度量单位说明(DIP,DP,PX,SP)
查看>>
Spring MVC和Struts2的比较的优点
查看>>
Redis配置文件redis.conf详解学习笔记八
查看>>
c++ qt 组播总结
查看>>
RobotFramework BuiltIn关键字笔记
查看>>
Spring整合JMS(四)——事务管理
查看>>
自己封装的golang 操作数据库方法
查看>>
Spring IOC启动流程学习(一)
查看>>
python tornado
查看>>
Android 自动换行的LinearLayout
查看>>
MacBook Pro电池校正
查看>>
初级python学习记录
查看>>